成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C# Attributes:定義設計期信息

開發 后端
C# Attributes是一種新的描述信息,在這篇指南中我們將明白怎么創建屬性并將其綁定至各種語言元素上,另外我們怎樣在運行時環境下獲取到attributes的一些信息。

約定:

     1.”attribute”和”attributes”均不翻譯

     2.”property”譯為“屬性”

     3. msdn中的原句不翻譯

     4.”program entity”譯為”語言元素”

C# Attributes介紹

Attributes是一種新的描述信息,我們既可以使用attributes來定義設計期信息(例如 幫助文件,文檔的URL),還可以用attributes定義運行時信息(例如,使XML中的元素與類的成員字段關聯起來)。我們也可以用attributes來創建一個“自描述”的組件。

C# Attributes定義

MSDN 中做如下定義(ms-help://MS.MSDNQTR.2002APR.1033/csspec/html/vclrfcsharpspec_17_2.htm)

"An attribute is a piece of additional declarative information that is specified for a declaration." 

使用預定義 Attributes

在c#中已有一小組預定義的attributes,在我們學習怎樣創建自定義attributes前,先來了解下在我們的代碼中使用那些預定義的attributes.

  1. using System;  
  2.  
  3. public class AnyClass   
  4.  
  5. {  
  6.     [Obsolete("Don't use Old method, use New method"true)]  
  7.  
  8.     static void Old( ) { }  
  9.  
  10.     static void New( ) { }  
  11.  
  12.     public static void Main( )   
  13.     {  
  14.         Old( );  
  15.     }  
  16. }  
  17.  

仔細看下該實例,在該實例中我們用到了”Obsolete”attribute,它標記了一個不該再被使用的語言元素(譯者注:這里的元素為方法),該屬性的***個參數是string類型,它解釋為什么該元素被荒棄,以及我們該使用什么元素來代替它。實際中,我們可以書寫任何其它文本來代替這段文本。第二個參數是告訴編譯器把依然使用這被標識的元素視為一種錯誤,這就意味著編譯器會因此而產生一個警告。

當我們試圖編譯上面的上面的程序,我們會得到如下錯誤:

AnyClass.Old()' is obsolete: 'Don't use Old method,  use New method'

開發自定義Attributes

現在我們即將了解怎么開發自定義的attributes。這兒有個小小處方,有它我們就可以學會創建自定義的attributes。

在C#中,我們的attribute類都派生于System.Attribute類 (A class that derives from the abstract class System.Attribute, whether directly or indirectly, is an attribute class. The declaration of an attribute class defines a new kind of attribute that can be placed on a declaration) ,我們就這么行動吧。

  1. using System;  
  2.  
  3. public class HelpAttribute : Attribute  
  4.  
  5. {  
  6.  
  7. }  

不管你是否相信我,就這樣我們就已經創建了一個自定義attribute。現在就可以用它來裝飾我們的類了,就像我們使用obsolete attribute一樣。

  1. [Help()]  
  2.  
  3. public class AnyClass  
  4.  
  5. {  
  6.  
  7. }  

注意:按慣例我們是用”Attribute“作為attribute類名的后綴,然而,當我們當我們把attribute綁定到某語言元素時,是不包含“Attribute“后綴的。編譯器首先在System.Attribute 的繼承類中查找該attribute,如果沒有找到,編譯器會把“Attribute“追加到該attribute的名字后面,然后查找它。

但是迄今為止,該attribute沒有任何用處。為了使它有點用處,讓我們在它里面加點東西吧。

  1. using System;  
  2. public class HelpAttribute : Attribute  
  3. {  
  4.     public HelpAttribute(String Descrition_in)  
  5.     {  
  6.         this.description = Description_in;  
  7.     }  
  8.     protected String description;  
  9.     public String Description   
  10.     {  
  11.         get   
  12.         {  
  13.             return this.description;  
  14.                    
  15.         }              
  16.     }      
  17. }  
  18. [Help("this is a do-nothing class")]  
  19. public class AnyClass  
  20. {  
  21. }  
  22.  

在上面的例子中,我們在attribute類中添加了一個屬性,在***一節中,我們將在運行時查詢該屬性。

定義或控制自定義Attribute的用法

AttributeUsage 類是另一預定義類(譯者注:attribute類本身用這個atrribute System.AttributeUsage來標記),它將幫助我們控制我們自定義attribute的用法,這就是,我們能為自定義的attribute類定義attributes。

它描述了一個自定義attribute類能被怎樣使用。

AttributeUsage 提供三個屬性,我們能將它們放置到我們的自定義attribute類上, ***個特性是:

ValidOn

通過這個屬性,我們能指定我們的自定義attribute可以放置在哪些語言元素之上。這組我們能把自定義attribute類放置其上的語言元素被放在枚舉器AttributeTargets 中。我們可以使用bitwise(譯者注:這個詞不知道怎么翻譯好,但他的意思是可以這么用:[AttributeUsage((AttributeTargets)4, AllowMultiple = false, Inherited = false )],4代表就是“class”元素,其它諸如1代表“assembly”,16383代表“all”等等)或者”.”操做符綁定幾個AttributeTargets 值。(譯者注:默認值為AttributeTargets.All)

AllowMultiple

該屬性標識我們的自定義attribte能在同一語言元素上使用多次。(譯者注:該屬性為bool類型,默認值為false,意思就是該自定義attribute在同一語言元素上只能使用一次)

Inherited

我們可以使用該屬性來控制我們的自定義attribute類的繼承規則。該屬性標識我們的自定義attribute是否可以由派生類繼承。((譯者注:該屬性為bool類型,默認值為false,意思是不能繼承)

讓我們來做點實際的東西吧,我們將把AttributeUsage attribute 放置在我們的help attribute 上并在它的幫助下,我們來控制help attribute的用法。

  1. using System;  
  2.  
  3. [AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = false )]  
  4.  
  5. public class HelpAttribute : Attribute  
  6.  
  7. {  
  8.  
  9.     public HelpAttribute(String Description_in)  
  10.  
  11.     {  
  12.  
  13.         this.description = Description_in;  
  14.  
  15.     }  
  16.  
  17.     protected String description;  
  18.  
  19.     public String Description  
  20.  
  21.     {  
  22.  
  23.         get   
  24.  
  25.         {  
  26.  
  27.             return this.description;  
  28.  
  29.         }              
  30.  
  31.     }      
  32.  
  33. }  
  34.  

首先我們注意 AttributeTargets.Class. 它規定這個help attribute 只能放置在語言元素”class”之上。這就意味著,下面的代碼將會產生一個錯誤。

AnyClass.cs: Attribute 'Help' is not valid on this declaration type.

It is valid on 'class' declarations only.

現在試著把它綁定到方法。

  1. [Help("this is a do-nothing class")]  
  2.  
  3. public class AnyClass  
  4.  
  5. {  
  6.  
  7.     [Help("this is a do-nothing method")]    //error  
  8.  
  9.     public void AnyMethod()  
  10.  
  11.     {  
  12.  
  13.     }  
  14.  
  15. }   
  16.  

我們可以使用 AttributeTargets.All 來允許 Help attribute 可以放置在任何預定義的語言元素上,那些可能的語言元素如下: 

可能的語言元素 

  1. [Help("this is a do-nothing class")]  
  2.  
  3. [Help("it contains a do-nothing method")]  
  4.  
  5. public class AnyClass  
  6.  
  7. {  
  8.  
  9.     [Help("this is a do-nothing method")]        //error  
  10.  
  11.     public void AnyMethod()  
  12.  
  13.     {  
  14.  
  15.     }  
  16.  
  17. }  

它產生了一個編譯錯誤:

AnyClass.cs: Duplicate 'Help' attribute

Ok!現在我們該討論下***那個屬性了,”Inherited”, 指出當把該attribute放置于一個基類之上,是否派生類也繼承了該attribute。如果綁定至某個attribute類的”Inherited”被設為true,那么該attribute就會被繼承,然而如果綁定至某個attribute類的”Inherited”被設為false或者沒有定義,那么該attribute就不會被繼承。

讓我們假設有如下的類關系。

  1. [Help("BaseClass")]   
  2.  
  3. public class Base  
  4.  
  5. {  
  6.  
  7. }  
  8.  
  9.    
  10.  
  11. public class Derive :  Base  
  12.  
  13. {  
  14.  
  15. }  
  16.  

我們有四種可能的綁定:

四種可能的綁定 

***種情況

如果我們查詢(我們將在后面來了解如何在運行時來查詢attributes)派生類中的help attribute,我們將不可能查詢到因為”Inherited”被設為了false。

第二種情況

第二種情況沒有什么不同,因為其”Inherited”也被設為了false。

第三種情況

為了解釋第三種和第四種情況,讓我們為派生類也綁定同一attribute。

  1. [Help("BaseClass")]   
  2.  
  3. public class Base  
  4.  
  5. {  
  6.  
  7. }  
  8.  
  9. [Help("DeriveClass")]   
  10.  
  11. public class Derive :  Base  
  12.  
  13. {  
  14.  
  15. }  
  16.  

現在我們查詢相關的help attribute ,我們將僅僅可以得到派生類的attribute,為什么這樣是因為help attribute雖然允許被繼承,但不能多次在同一語言元素上使用,所以基類中的help attribute被派生類的help attribute 重寫了。

第四種情況

在第四種情況中,當我們查詢派生類的help attribute 時,我們可以得到兩個attributes,當然是因為help attribute既允許被繼承,又允許在同一語言元素上多次使用的結果。

注意:AttributeUsage attribute 僅應用在那種是System.Attribute 派生的attriubte類而且綁定值該attriubte類的AllowMultiple和Inherited均為false上才是有效的。

可選參數 vs. 命名參數

可選參數是attribute類構造函數的參數。它們是強制的,必須在每次在attribute綁定至某語言元素時提供一個值。而另一方面,命名參數倒是真正的可選參數,不是在attribute構造函數的參數。

為了更加詳細的解釋,讓我們在Help類中添加另外的屬性。

  1. [AttributeUsage(AttributeTargets.Class, AllowMultiple = false,  
  2.  
  3.  Inherited = false)]  
  4.  
  5. public class HelpAttribute : Attribute  
  6.  
  7. {  
  8.  
  9.     public HelpAttribute(String Description_in)  
  10.  
  11.     {  
  12.  
  13.         this.description = Description_in;  
  14.  
  15.         this.verion = "No Version is defined for this class";  
  16.  
  17.     }  
  18.  
  19.     protected String description;  
  20.  
  21.     public String Description  
  22.  
  23.     {  
  24.  
  25.         get   
  26.  
  27.         {  
  28.  
  29.             return this.description;  
  30.  
  31.         }  
  32.  
  33.     }  
  34.  
  35.     protected String version;  
  36.  
  37.     public String Version  
  38.  
  39.     {  
  40.  
  41.         get   
  42.  
  43.         {  
  44.  
  45.             return this.version;  
  46.  
  47.         }  
  48.  
  49.         //if we ever want our attribute user to set this property,   
  50.  
  51.         //we must specify set method for it   
  52.  
  53.         set   
  54.  
  55.         {  
  56.  
  57.             this.verion = value;  
  58.  
  59.         }  
  60.  
  61.     }  
  62.  
  63. }  
  64.  
  65. [Help("This is Class1")]  
  66.  
  67. public class Class1  
  68.  
  69. {  
  70.  
  71. }  
  72.  
  73.  
  74.  
  75. [Help("This is Class2", Version = "1.0")]  
  76.  
  77. public class Class2  
  78.  
  79. {  
  80.  
  81. }  
  82.  
  83.  
  84.  
  85. [Help("This is Class3", Version = "2.0",   
  86.  
  87.  Description = "This is do-nothing class")]  
  88.  
  89. public class Class3  
  90.  
  91. {  
  92.  
  93. }  
  94.  

當我們在Class1中查詢Help attribute已經它的屬性,我們將得到:

Help.Description : This is Class1

Help.Version :No Version is defined for this class

因為我們沒有為Version這個屬性定義任何任何值,所以在構造函數中設定的值被我們查詢出來了。如果沒有定義任何值,那么就會賦一個該類型的默認值(例如:如果是int型,默認值就是0)。

現在,查詢Class2的結果是:

Help.Description : This is Class2

Help.Version :  1.0

我們不能為了可選參數而使用多個構造函數,應該用已命名參數來代替。我們之所以稱它們為已命名的,是因為當我們在構造函數為它們提供值時,我們必須命名它們。例如,在第二個類中,我們如是定義Help。

[Help("This is Class2", Version = "1.0")]

在 AttributeUsage 例子中, 參數”ValidOn”是可選參數,而“Inherited“和“AllowMultiple“ 是命名參數。

注意:為了在attribute的構造函數中設定命名參數的值,我們必須為相應的屬性提供一個set方法否則會引起編譯期錯誤:

'Version' : Named attribute argument can't be a read only property

現在,我們在Class3中查找Help attribute 及其屬性會發生什么呢?結果是跟上面提到的相同的編譯期錯誤。

'Desciption' : Named attribute argument can't be a read only property

現在我們修改下Help類,為屬性”Description”加一個set方法。現在的輸出就是:

Help.Description : This is do-nothing class

Help.Version : 2.0

在屏幕后面究竟發生了什么呢?首先帶有可選參數的構造函數被調用,然后,每個命名參數的set方法被調用,在構造函數中賦給命名參數的值被set方法所覆寫。

參數類型

一個attribute類的參數類型被限定在如下類型中:

attribute類的參數類型  

Attributes 標記

假設,我們想把Help attribute 綁定至元素 assembly。***個問題是我們要把Help attribute 放在哪兒才能讓編譯器確定該attribute是綁定至整個assembly呢?考慮另一種情況,我們想把attribute綁定至一個方法的返回類型上,怎樣才能讓編譯器確定我們是把attribute綁定至方法的返回類型上,而不是整個方法呢?

為了解決諸如此類的含糊問題,我們使用attribute標識符,有了它的幫助,我們就可以確切地申明我們把attribute 綁定至哪一個語言元素。

例如:

[assembly: Help("this a do-nothing assembly")]

這個在Help attribute 前的assembly標識符確切地告訴編譯器,該attribute被綁定至整個assembly。可能的標識符有: 

可能的標識符 

在運行時查詢Attributes

現在我們明白怎么創建attribtes和把它們綁定至語言元素。是時候來學習類的使用者該如何在運行時查詢這信息。

為了查詢一語言元素上綁定的attributes,我們必須使用反射。反射有能力在運行時發現類型信息。

我們可以使用.NET Framework Reflection APIs 通過對整個assembly元數據的迭代,列舉出assembly中所有已定義的類,類型,還有方法。

記住那舊的Help attribute 和AnyClass 類。

  1. using System;  
  2.  
  3. using System.Reflection;  
  4.  
  5. using System.Diagnostics;  
  6.  
  7.  
  8.  
  9. //attaching Help attribute to entire assembly  
  10.  
  11. [assembly : Help("This Assembly demonstrates custom attributes   
  12.  
  13.  creation and their run-time query.")]  
  14.  
  15.  
  16.  
  17. //our custom attribute class  
  18.  
  19. public class HelpAttribute : Attribute  
  20.  
  21. {  
  22.  
  23.     public HelpAttribute(String Description_in)  
  24.  
  25.     {  
  26.  
  27.         //  
  28.  
  29.         // TODO: Add constructor logic here  
  30.  
  31.         this.description = Description_in;  
  32.  
  33.         //  
  34.  
  35.     }  
  36.  
  37.     protected String description;  
  38.  
  39.     public String Description  
  40.  
  41.     {  
  42.  
  43.         get   
  44.  
  45.         {  
  46.  
  47.             return this.deescription;  
  48.  
  49.                    
  50.  
  51.         }              
  52.  
  53.     }      
  54.  
  55. }  
  56.  
  57. //attaching Help attribute to our AnyClass  
  58.  
  59. [HelpString("This is a do-nothing Class.")]  
  60.  
  61. public class AnyClass  
  62.  
  63. {  
  64.  
  65. //attaching Help attribute to our AnyMethod  
  66.  
  67.     [Help("This is a do-nothing Method.")]  
  68.  
  69.     public void AnyMethod()  
  70.  
  71.     {  
  72.  
  73.     }  
  74.  
  75. //attaching Help attribute to our AnyInt Field  
  76.  
  77.     [Help("This is any Integer.")]  
  78.  
  79.     public int AnyInt;  
  80.  
  81. }  
  82.  
  83. class QueryApp  
  84.  
  85. {  
  86.  
  87.     public static void Main()  
  88.  
  89.     {  
  90.  
  91.     }  
  92.  
  93. }  
  94.  

我們將在接下來的兩節中在我們的Main方法中加入attribute查詢代碼。

查詢程序集的Attributes

在接下來的代碼中,我們先得到當前的進程名稱,然后用Assembly類中的LoadForm()方法加載程序集,再有用GetCustomAttributes()方法得到被綁定至當前程序集的自定義attributes,接下來用foreach語句遍歷所有attributes并試圖把每個attribute轉型為Help attribute(即將轉型的對象使用as關鍵字有一個優點,就是當轉型不合法時,我們將不需擔心會拋出異常,代之以空值(null)作為結果),接下來的一行就是檢查轉型是否有效,及是不是為空,跟著就顯示Help attribute的“Description”屬性。

  1. class QueryApp  
  2.  
  3. {  
  4.  
  5.     public static void Main()  
  6.  
  7.     {  
  8.  
  9.         HelpAttribute HelpAttr;  
  10.  
  11.  
  12.         //Querying Assembly Attributes  
  13.  
  14.         String assemblyName;  
  15.  
  16.         Process p = Process.GetCurrentProcess();  
  17.  
  18.         assemblyName = p.ProcessName + ".exe";  
  19.  
  20.  
  21.         Assembly a = Assembly.LoadFrom(assemblyName);  
  22.  
  23.  
  24.         foreach (Attribute attr in a.GetCustomAttributes(true))  
  25.         {  
  26.  
  27.             HelpAttr = attr as HelpAttribute;  
  28.  
  29.             if (null != HelpAttr)  
  30.  
  31.             {  
  32.  
  33.                 Console.WriteLine("Description of {0}:\n{1}",   
  34.  
  35.                                   assemblyName,HelpAttr.Description);  
  36.  
  37.             }  
  38.  
  39.         }  
  40.  
  41. }  
  42.  
  43. }  
  44.  

程序輸出如下:

Description of QueryAttribute.exe:

This Assembly demonstrates custom attributes creation and

their run-time query.

Press any key to continue

查詢類、方法、類成員的Attributes

下面的代碼中,我們惟一不熟悉的就是Main()方法中的***行。

Type type = typeof(AnyClass);

它用typeof操作符得到了一個與我們AnyClass類相關聯的Type型對象。剩下的查詢類attributes代碼就與上面的例子是相似的,應該不要解釋了吧(我是這么想的)。

為查詢方法和類成員的attributes,首先我們得到所有在類中存在的方法和成員,然后我們查詢與它們相關的所有attributes,這就跟我們查詢類attributes一樣的方式。

  1. class QueryApp  
  2.  
  3. {  
  4.  
  5.     public static void Main()  
  6.  
  7.     {  
  8.  
  9.  
  10.  
  11.         Type type = typeof(AnyClass);  
  12.  
  13.         HelpAttribute HelpAttr;  
  14.  
  15.  
  16.  
  17.  
  18.  
  19.         //Querying Class Attributes  
  20.  
  21.         foreach (Attribute attr in type.GetCustomAttributes(true))  
  22.  
  23.         {  
  24.  
  25.             HelpAttr = attr as HelpAttribute;  
  26.  
  27.             if (null != HelpAttr)  
  28.  
  29.             {  
  30.  
  31.                 Console.WriteLine("Description of AnyClass:\n{0}",   
  32.  
  33.                                   HelpAttr.Description);  
  34.  
  35.             }  
  36.  
  37.         }  
  38.  
  39.         //Querying Class-Method Attributes    
  40.  
  41.         foreach(MethodInfo method in type.GetMethods())  
  42.  
  43.         {  
  44.  
  45.             foreach (Attribute attr in method.GetCustomAttributes(true))  
  46.  
  47.             {  
  48.  
  49.                 HelpAttr = attr as HelpAttribute;  
  50.  
  51.                 if (null != HelpAttr)  
  52.  
  53.                 {  
  54.  
  55.                     Console.WriteLine("Description of {0}:\n{1}",   
  56.  
  57.                                       method.Name,   
  58.  
  59.                                       HelpAttr.Description);  
  60.  
  61.                 }  
  62.  
  63.             }  
  64.  
  65.         }  
  66.  
  67.         //Querying Class-Field (only public) Attributes  
  68.  
  69.         foreach(FieldInfo field in type.GetFields())  
  70.  
  71.         {  
  72.  
  73.             foreach (Attribute attr in field.GetCustomAttributes(true))  
  74.  
  75.             {  
  76.  
  77.                 HelpAttr= attr as HelpAttribute;  
  78.  
  79.                 if (null != HelpAttr)  
  80.  
  81.                 {  
  82.  
  83.                     Console.WriteLine("Description of {0}:\n{1}",  
  84.  
  85.                                       field.Name,HelpAttr.Description);  
  86.  
  87.                 }  
  88.  
  89.             }  
  90.  
  91.         }  
  92.  
  93.     }  
  94.  
  95. }  
  96.  

以上就介紹了C# Attributes的一些使用方法。

【編輯推薦】

責任編輯:book05 來源: cnblogs
相關推薦

2024-11-14 00:09:57

C# 特性信息CLR

2009-08-07 17:57:26

C#定義事件應用

2009-08-25 16:24:44

C#信息架構視圖

2009-08-26 10:24:04

C# Observer

2009-08-27 17:59:56

C#接口定義

2009-08-07 13:39:13

C#定義整型數組

2009-08-10 14:55:04

C#定義Nullabl

2009-08-31 16:47:39

C#接口的定義

2009-08-03 18:26:18

C#定義接口成員

2009-08-18 17:51:17

C#實現Interne

2009-09-02 17:53:42

C#程序設計Windows窗體

2009-09-02 16:23:27

C# Singleto

2009-09-07 06:07:46

C#窗體設計

2009-08-12 15:55:12

C#接口定義

2009-09-28 14:45:22

C#接口的定義

2009-09-24 15:20:54

C#接口定義

2009-09-10 17:48:05

C# button

2009-08-13 14:46:03

C#結構體定義

2009-08-04 08:58:01

C#自定義特性

2009-08-25 10:44:08

C#接口定義接口
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久精品国产亚洲 | 婷婷免费在线 | 91色在线| 北条麻妃一区二区三区在线观看 | 伊人99| 成人夜晚看av | 日韩欧美在线观看视频网站 | 久久精彩视频 | 精品欧美黑人一区二区三区 | www.婷婷| 亚洲va中文字幕 | 国产精品自产av一区二区三区 | 久久久99精品免费观看 | 9色视频在线| 91精品一区 | 久久蜜桃av一区二区天堂 | 羞羞网站免费 | 欧美精品乱码99久久影院 | 欧美色性 | 91久久国产综合久久 | 日韩中文字幕视频在线观看 | 国产高清免费 | 一级二级三级黄色 | 欧美日本一区二区 | 国产精品日本一区二区在线播放 | 精品福利一区二区三区 | 中文字幕精品一区 | 亚洲免费在线播放 | 国产一区二区激情视频 | 草久免费视频 | 亚洲三区视频 | 亚洲精品国产电影 | 国产精品夜夜夜一区二区三区尤 | 久久精品亚洲精品国产欧美 | 日日骚网| 毛片99 | 小h片免费观看久久久久 | 91久久久久久久久 | 天堂中文资源在线 | 国产成人午夜电影网 | 国产精彩视频 |