日期:2014-05-17  浏览次数:20864 次

[转载]C# 特性(Attribute)入门教程
特性(Attributes)是一种崭新的声明性信息。我们不仅可以通过特性来定义设计层面的信息(例如help file, URL for documentation)以及运行时(run-time)信息(例如使XML与class相联系),而且我们还可以利用特性建立自描述(self-describing)组件。在这篇教程中,我们将会看到如何建立和添加特性到各种程序实体以及如何在运行时环境中获取特性信息。

定义

  正如MSDN中所描述的那样-----
  
    “特性是被指定给某一声明的一则附加的声明性信息。”

使用预定义(Pre-defined)特性
  
  在C#中,有一个小的预定义特性集合。在学习如何建立我们自己的定制特性(custom attributes)之前,我们先来看看在我们的代码中如何使用预定义特性。

using System; 
  public class AnyClass 
   { 
    [Obsolete("Don't use Old method, use New method", true)] 
   static void Old( ) { } 
   
   static void New( ) { } 
   
   public static void Main( ) 
    { 
       Old( ); 
    } 
   }


  我们先来看一下上面这个例子,在这个例子中我们使用了Obsolete特性,它标记了一个不应该再被使用的程序实体。第一个参数是一个字符串,它解释了为什么该实体是过时的以及应该用什么实体来代替它。实际上,你可以在这里写任何文本。第二个参数告诉编译器应该把使用这个过时的程序实体当作一种错误。它的默认值是false,也就是说编译器对此会产生一个警告。
  
  当我们尝试编译上面这段程序的时候,我们将会得到一个错误:
  
AnyClass.Old()' is obsolete: 'Don't use Old method, use New method'   
开发定制特性(custom attributes)
  
  现在让我们来看看如何开发我们自己的特性。
  
  首先我们要从System.Attribute派生出我们自己的特性类(一个从System.Attribute抽象类继承而来的类,不管是直接还是间接继承,都会成为一个特性类。特性类的声明定义了一种可以被放置在声明之上新的特性)。

using System; 
  public class HelpAttribute : Attribute 
   { 
   } 

  不管你是否相信,我们已经建立了一个定制特性,现在我们可以用它来装饰现有的类就好像上面我们使用Obsolete attribute一样。

[Help()] 
  public class AnyClass 
   { 
   }


  注意:对一个特性类名使用Attribute后缀是一个惯例。然而,当我们把特性添加到一个程序实体,是否包括Attribute后缀是我们的自由。编译器会首先在System.Attribute的派生类中查找被添加的特性类。如果没有找到,那么编译器会添加Attribute后缀继续查找。

  到目前为止,这个特性还没有起到什么作用。下面我们来添加些东西给它使它更有用些。

using System; 
  public class HelpAttribute : Attribute 
   { 
   public HelpAttribute(String Descrition_in) 
    { 
     this.description = Description_in; 
    } 
   protected String description; 
   public String Description 
    { 
   get 
    { 
     return this.description; 
    } 
    } 
   } 
   [Help("this is a do-nothing class")] 
  public class AnyClass 
   { 
   } 



  在上面的例子中,我们给HelpAttribute特性类添加了一个属性并且在后续的部分中我们会在运行时环境中查寻它。


定义或控制特性的使用

  AttributeUsage类是另外一个预定义特性类,它帮助我们控制我们自己的定制特性的使用。它描述了一个定制特性如和被使用。
  
  AttributeUsage有三个属性,我们可以把它放置在定制属性前面。第一个属性是:

   ValidOn
  
  通过这个属性,我们能够定义定制特性应该在何种程序实体前放置。一个属性可以被放置的所有程序实体在AttributeTargets enumerator中列出。通过OR操作我们可以把若干个AttributeTargets值组合起来。

  AllowMultiple
  
  这个属性标记了我们的定制特性能否被重复放置在同一个程序实体前多次。

  Inherited
  
  我们可以使用这个属性来控制定制特性的继承规则。它标记了我们的特性能否被继承。

  下面让我们来做一些实际的东西。我们将会在刚才的Help特性前放置AttributeUsage特性以期待在它的帮助下控制Help特性的使用。

using System; 
   [AttributeUsage(AttributeTargets.Class), AllowMultiple = false, 
    Inherited = false ] 
  public class HelpAttribute : Attribute 
   { 
   public HelpAttribute(String Description_in) 
    { 
     this.description = Description_in; 
    } 
   protected String description; 
   public String Description 
    { 
   get 
    { 
     return this.description; 
    } 
    } 
   }



  先让我们来看一下AttributeTargets.Class。它规定了Help特性只能被放在class的前面。这也就意味着下面的代码将会产生错误:

[Help("this is a do-nothing class")] 
  public class AnyClass 
   { 
    [Help("this is a do-nothing method")] //error 
   public void AnyMethod() 
    { 
    } 
   } 



  编译器报告错误如下:
  
AnyClass.cs: Attribute 'Help' is not valid on this declaration type.
  
It is valid on 'class' declarations only.

  我们可以使用AttributeTargets.All来允许Help特性被放置在任何程序实体前。可能的值是:

Assembly,
&n