在.NET运行时了解类型信息(3)

访问自定义属性

当属性与程序元素相关联后,可以使用反射来查询它们是否存在以及它们的值。用于查询属性的主要反射方法包含在 System.Reflection.MemberInfo.GetCustomAttributes 和 System.Reflection.Assembly.GetCustomAttributes 中。

自定义属性的可访问性根据附加该属性的程序集来进行检查。这相当于检查附加自定义属性的程序集中的类型上的方法是否可以调用自定义属性的构造函数。

诸如 System.Reflection.Assembly.GetCustomAttributes(Type, Boolean) 等方法检查类型参数的可见性和可访问性。只有包含用户定义类型的程序集中的代码才能使用 GetCustomAttributes 检索该类型的自定义属性。

以下代码示例是典型的自定义属性设计模式。它说明运行库自定义属性反射模型。

[C#]

System.DLL

public class DescriptionAttribute : Attribute

{

}

System.Web.DLL

internal class MyDescriptionAttribute : DescriptionAttribute

{

}

public class LocalizationExtenderProvider

{

[MyDescriptionAttribute(...)]

public CultureInfo GetLanguage(...)

{

}

}

如果试图为附加到 ** GetLanguage ** 方法的公共自定义属性类型 DescriptionAttribute 检索自定义属性,运行库将执行以下操作:

  1. 运行库检查 ** Type.GetCustomAttributes ** (Type type ) 的 DescriptionAttribute 类型参数是否为公共的,并检查其是否可见或可以访问。
  2. 运行库检查从 ** DescriptionAttribute ** 导出的用户定义类型 MyDescriptionAttributeSystem.Web.DLL 程序集(它在该程序集中附加到 GetLanguage () 方法)内是否可见和可以访问。
  3. 运行库检查 ** MyDescriptionAttribute ** 的构造函数是否在 System.Web.DLL 程序集中可见和可以访问。
  4. 运行库调用带有自定义属性参数的 ** MyDescriptionAttribute ** 的构造函数,然后将新对象返回给调用方。

自定义属性反射模型可能会在定义类型的程序集外泄漏用户定义类型的实例。这与运行库系统库中返回用户定义类型的实例的成员(例如返回 ** RuntimeMethodInfo ** 对象数组的 Type.GetMethods())相同。为了防止客户端发现关于用户定义的自定义属性类型的信息,请将该类型的成员定义为非公共成员。

以下代码示例说明使用反射访问自定义属性的基本方法。

[C#]

class MainClass

{

public static void Main()

{

System.Reflection.MemberInfo info = typeof(MyClass);

object[] attributes = info.GetCustomAttributes();

for (int i = 0; i < attributes.Length; i ++)

{

System.Console.WriteLine(attributes[i]);

}

}

}

指定完全限定的类型名称

要为各种反射操作提供有效的输入,必须指定类型名称。完全限定的类型名称包含程序集名称指定、命名空间指定和类型名称。类型名称指定将由 Type.GetType、Module.GetType、ModuleBuilder.GetType 和 Assembly.GetType 来使用。

类型名称的 Backus-Naur 形式语法

Backus-Naur 形式 (BNF) 定义正式语言的语法。下表中的 BNF 词法规则将说明如何识别有效的输入。最终元素(无法再减小的元素)将全部以大写字母显示。非最终元素(可以再减小的元素)则显示为大小写混合或带单引号的字符串,但单引号 (') 不是语法本身的一部分。管道字符 (|) 表示具有子规则的规则。

** 完全限定类型名称的 BNF 语法 ** ** **


TypeSpec := ReferenceTypeSpec

| SimpleTypeSpec

ReferenceTypeSpec := SimpleTypeSpec '&'

SimpleTypeSpec := PointerTypeSpec

| ArrayTypeSpec

| TypeName

PointerTypeSpec := SimpleTypeSpec '*'

ArrayTypeSpec := SimpleTypeSpec '[ReflectionDimension]'

| SimpleTypeSpec '[ReflectionEmitDimension]'

ReflectionDimension := '*'

| ReflectionDimension ',' ReflectionDimension

| NOTOKEN

ReflectionEmitDimension := '*'

| Number '..' Number

| Number '...'

| ReflectionDimension ',' ReflectionDimension

| NOTOKEN

Number := [0-9]+

TypeName := NamespaceTypeName

| NamespaceTypeName ',' AssemblyNameSpec

NamespaceTypeName := NestedTypeName

| NamespaceSpec '.'NestedTypeName

NestedTypeName := IDENTIFIER

| NestedTypeName '+' IDENTIFIER

NamespaceSpec := IDENTIFIER

| NamespaceSpec '.'IDENTIFIER

AssemblyNameSpec := IDENTIFIER

| IDENTIFIER ',' AssemblyProperties

AssemblyProperties := AssemblyProperty

| AssemblyProperties ',' AssemblyProperty

AssemblyProperty := AssemblyPropertyName '=' AssemblyPropertyValue

指定特殊字符

在 TypeName 中,IDENTIFIER 是由语言规则所确定的任何有效名称。

反斜杠 (\) 可用作转义符来分隔以下用作 IDENTIFIER 一部分的标记。

** 标记 ** ** **

|

** 含义 ** ** **

---|---

,

|

程序集分隔符。

\+

|

嵌套类型分隔符。

&amp;

|

引用类型。

\*

|

指针类型。

\[

|

数组维度分隔符。

\]

|

数组维度分隔符。

\.

|

只有在数组指定中使用句点时,才应在句点前使用反斜杠。 NamespaceSpec 中的句点不采用反斜杠。

\\

|

用作字符串的反斜杠。

请注意,在除 AssemblyNameSpec 之外的所有 TypeSpec 组成部分中,空格都是相关的。在 AssemblyNameSpec 中,“,”分隔符之前的空格相关,但“,”分隔符之后的空格将被忽略。

反射类(如 Type.FullName)将返回经过处理的名称,以便使返回的名称可以在对 GetType 的调用中使用(例如在 MyType.GetType(myType.FullName) 中)。

例如,某个类型的完全限定名称可能是 Ozzy.OutBack.Kangaroo+Wallaby,MyAssembly

如果命名空间为 Ozzy.Out+Back ,则必须在加号前加反斜杠。否则,分析器会将其解释为嵌套分隔符。反射会将该字符串当作 Ozzy.Out\+Back.Kangaroo+Wallaby,MyAssembly 发出。

指定程序集名称

程序集名称指定中所需的最少信息为程序集的文本名称 (IDENTIFIER)。您可以在 IDENTIFIER 后添加下表所述的逗号分隔属性/值对列表。IDENTIFIER 命名应遵循文件命名的规则。IDENTIFIER 不区分大小写。

** 有效的程序集属性 **

** 属性名称 ** ** **

|

** 说明 ** ** **

|

** 允许值 ** ** **

---|---|---

** Version **

|

程序集版本号

|

_ Major.Minor.Build.Revision _ ,其中 _ Major _ 、 _ Minor _ 、 _ Build _ 和 Revision 是 0 和 65535 之间(含 0 和 65535)的整数。

** PublicKey **

|

完全公钥

|

完全公钥的十六进制字符串值。指定 ** null ** (在 Microsoft Visual Basic .NET 中为 Nothing )可显式地指示私有程序集。

** PublicKeyToken **

|

公钥标记(完全公钥的 8 字节哈希)

|

公钥标记的十六进制字符串值。指定 ** null ** (在 Visual Basic .NET 中为 Nothing )可显式地指示私有程序集。

** Culture **

|

程序集区域性

|

程序集的 RFC-1766 格式区域性,或者对于独立于语言(非附属)的程序集为“非特定”。

** Custom **

|

自定义的二进制大对象 (BLOB)。它当前仅用于由本机图像生成器 (Ngen.exe) 生成的程序集。

|

自定义的字符串,由本机图像生成器工具用来向程序集缓存通知所安装的程序集为本机图像,因此将安装在本机图像缓存中。也称作 Zap 字符串。

以下代码示例显示带有默认区域性且名称简单的程序集的 ** AssemblyName ** 。

[C#]


com.microsoft.crypto, Culture="" 

以下代码示例显示区域性为 “en”且带有强名称的程序集的完全限定引用。

[C#]


com.microsoft.crypto, Culture=en, PublicKeyToken=a5d015c7d5a0b012,


   Version=1.0.0.0 

以下代码示例显示部分指定的 ** AssemblyName ** ,它可以由带有强名称或简单名称的程序集来满足。

[C#]


com.microsoft.crypto


com.microsoft.crypto, Culture=""


com.microsoft.crypto, Culture=en 

以下每个代码示例都显示一个部分指定的 ** AssemblyName ** ,它必须由带有简单名称的程序集来满足。

[C#]


com.microsoft.crypto, Culture="", PublicKeyToken=null 


com.microsoft.crypto, Culture=en, PublicKeyToken=null

以下每个代码示例都显示一个部分指定的 ** AssemblyName ** ,它必须由带有强名称的程序集来满足。

[C#]


&lt;
Published At
Categories with Web编程
Tagged with
comments powered by Disqus