特性Attribute概述

特性(Attribute)是一种特殊的类型,可以加载到程序集或者程序集的类型上,这些类型包括模块、类、接口、结构、构造函数、方法、字段等,加载了特性的类型称之为特性的目标。这里为与属性(Property)区分,所以称之为特性(Attribute)。特性是为程序集添加元数据的一种机制,通过它可以为编译器提供指示或者对数据进行说明。例如前段时间学习的Remoting技术(主要用于应用程序域之间的对象通信)中在应用程序域间的引用对象时该对象具有序列化(Serializable)这个特性。下面使用ObsoleteAttribute特性学习特性的使用方法。

System.ObsoleteAttribute实例

我们有一个旧的方法SendMsg()方法由于功能和效率上的优化重载了这个方法,需要将原来的方法加上Obsolete特性告诉编译器这个方法已经过时,然后编译器发现程序中有地方使用该特性标记过的方法时,就会给出一个如下所示警告信息:

测试代码如下所示:

using System;

namespace AttributeTest
{
    /// <summary>
    /// 信息实体类
    /// </summary>
    public class Message
    {
        //此处具体实现略
    }
    /// <summary>
    /// 信息操作类
    /// </summary>
    public class MessageOperation
    {
        [Obsolete("请使用新的SendMsg(Message msg)重载方法")]
        public static void SendMsg()
        {
            Console.WriteLine("这是旧的SendMsg方法");
        }
        public static void SendMsg(Message msg)
        {
            Console.WriteLine("这是新的SendMsg方法");
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            //使用旧的方法SendMsg()
            MessageOperation.SendMsg();
            //使用新的方法SendMsg(Message msg)
            MessageOperation.SendMsg(new Message());
        }
    }
}

这样一来,开发人员编译运行时就看到了"请使用新的SendMsg(Message msg)重载方法"警告,然后就知道应该选用新的SendMsg(Message msg)重载方法。通过上面的例子可以看到特性使用的方法:首先是一对方括号“[]”,在左方括号中后紧跟特性的名称,比如Obsolete。随后是一个圆括号“()”,在这个圆括号中,不光可以传入构造函数的参数,还可以向特性的属性赋值。在Obsolete的例子中,仅传递了构造函数的参数。构造参数又称为位置参数(传入顺序必须与构造函数声明时一致);属性参数也叫做命名参数。

下面通过自定义特性进一步学习特性。

自定义特性

假设我们有这么一个需求:创建或者更新一个类文件是需要说明这个类是什么时候谁创建的,以后是谁更新的等,是不是想下面这样在类上添加注释:

这样手动添加是可以记录下来但是需要查看所有类型的更新记录显然就不适合了。带着这么一个悬念,我们先看看Obsolete特性的具体定义如下所示:

using System.Runtime.InteropServices;

namespace System
{
    // 摘要:
    //     标记不再使用的程序元素。无法继承此类。
    [Serializable]
    [AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct | AttributeTargets.Enum | AttributeTargets.Constructor | AttributeTargets.Method | AttributeTargets.Property | AttributeTargets.Field | AttributeTargets.Event | AttributeTargets.Interface | AttributeTargets.Delegate, Inherited = false)]
    [ComVisible(true)]
    public sealed class ObsoleteAttribute : Attribute
    {
        // 摘要:
        //     使用默认属性初始化 System.ObsoleteAttribute 类的新实例。
        public ObsoleteAttribute();
        //
        // 摘要:
        //     使用指定的变通方法消息初始化 System.ObsoleteAttribute 类的新实例。
        //
        // 参数:
        //   message:
        //     描述可选的变通方法的文本字符串。
        public ObsoleteAttribute(string message);
        //
        // 摘要:
        //     使用变通方法消息和布尔值初始化 System.ObsoleteAttribute 类的新实例,该布尔值指示是否将使用已过时的元素视为错误。
        //
        // 参数:
        //   message:
        //     描述可选的变通方法的文本字符串。
        //
        //   error:
        //     指示是否将使用已过时的元素视为错误的布尔值。
        public ObsoleteAttribute(string message, bool error);

        // 摘要:
        //     获取指示编译器是否将使用已过时的程序元素视为错误的布尔值。
        //
        // 返回结果:
        //     如果将使用已过时的元素视为错误,则为 true;否则为 false。默认为 false。
        public bool IsError { get; }
        //
        // 摘要:
        //     获取变通方法消息,包括对可选程序元素的说明。
        //
        // 返回结果:
        //     变通方法文本字符串。
        public string Message { get; }
    }
}

我们看到Obsolete特性在定义时继承了Attribute(这是特性必需的),使用了Serializable、ComVisible和AttributeUsage这三个属性。Serializable属性表明类型支持序列化;ComVisible为 true,指示该托管类型(public 类型)对 COM 是可见的;AttributeUsage定义您自己的特性类时,可通过在特性类上放置 AttributeUsageAttribute 来控制特性类的使用方式。

通过上述的学习并根据前面的需求,我们自定义一个特性RecordAttribute,实现代码如下所示:

//该特性可用于方法和类并且可以重复的特价到一个类型上
    [AttributeUsage(AttributeTargets.Class|AttributeTargets.Method,AllowMultiple = true)]
    public class RecordAttribute : Attribute
    {
        private string _recordType;//记录类型:更新or创建
        private string _author;//作者
        private string _date;//日期
        private string _comment;//备注
        //构造函数
        public RecordAttribute(string recordType, string author, string date)
        {
            _recordType = recordType;
            _author = author;
            _date = date;
        }
        //对于位置参数,通常只提供get访问器

        public string RecordType { get { return _recordType; }}
        public string Author { get { return _author; } }
        public string Date { get { return _date; } }
        //构建一个属性,在特性中也叫做命名参数
        public string Comment { get; set; }
    }

记录类修改或者更新的特性创建好了,下面是使用示例:

[Record("更新", "鞠小军", "2014.8.3", Comment = "添加ToString()方法")]
    [Record("更新", "鞠小军", "2014.8.3")]
    [Record("创建","鞠小军","2014.8.2")]
    public class Message
    {
        //此处具体实现略
        public override string ToString()
        {
            return "添加了ToString()方法";
        }
    }

这样就成功实现了上面的需求,Message类添加的Record属性实际上作为元数据添加到了程序集中。

.Net内置特性Attribute介绍的更多相关文章

  1. python内置函数详细介绍

    知识内容: 1.python内置函数简介 2.python内置函数详细介绍 一.python内置函数简介 python中有很多内置函数,实现了一些基本功能,内置函数的官方介绍文档:    https: ...

  2. linux awk 内置函数详细介绍(实例)

    这节详细介绍awk内置函数,主要分以下3种类似:算数函数.字符串函数.其它一般函数.时间函数 一.算术函数: 以下算术函数执行与 C 语言中名称相同的子例程相同的操作: 函数名 说明 atan2( y ...

  3. Python中内置函数的介绍

    内置函数的功能介绍 常用内置函数如下: 1.abs() 绝对值 格式:abs(x) 例如:print(abs(-18)) >>> 18 返回值:number #该函数主要用于数值类的 ...

  4. 类的内置方法__attr__介绍

    1.hasattr  getaddr  setaddr delattr 这四个函数同样也适用于类 class BlackMedium: feture="Ugly" def __in ...

  5. mysql 内置功能 存储过程介绍

    存储过程介绍 就是mysql内置功能把逻辑写好 的功能给封装好,封装成一个接口名,把接口名丢给应用程序,应用程序直接调用接口名实现一系列增删改查功能 这个接口叫存储过程 基于存储过程封装成一个功能 存 ...

  6. linux awk 内置函数详细介绍(实例)

    这节详细介绍awk内置函数,主要分以下3种类似:算数函数.字符串函数.其它一般函数.时间函数 一.算术函数: 以下算术函数执行与 C 语言中名称相同的子例程相同的操作: 函数名 说明 atan2( y ...

  7. Linux基础教程 linux awk内置变量使用介绍

    awk是个优秀文本处理工具,可以说是一门程序设计语言.下面是兄弟连Linux培训 给大家介绍的awk内置变量. 一.内置变量表 属性 说明 $0 当前记录(作为单个变量) $1~$n 当前记录的第n个 ...

  8. JSP第二篇【内置对象的介绍、4种属性范围、应用场景】

    什么是JSP内置对象 JSP引擎在调用JSP对应的jspServlet时,会传递或创建9个与web开发相关的对象供jspServlet使用.JSP技术的设计者为便于开发人员在编写JSP页面时获得这些w ...

  9. Orchard内置特性(以模块来说的)

    本文链接:http://www.cnblogs.com/souther/p/4539169.html 主目录 Orchard中有很多可以直接和多次使用的特性,这些东西在官方的Gallery中可以找到. ...

随机推荐

  1. Sprint(第九天11.22)

  2. avalon复杂绑定

    样式操作:ms-css-样式名=“样式值”,ms-class ms-css-width="prop"(自动补px) ms-css-height="{{prop}}%&qu ...

  3. 图片懒加载--判断div ul中的li是否已经滑动到可视区域里

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name ...

  4. rxjava源码中的线程知识

    rxjava源码中的线程知识 rx的最精简的总结就是:异步 这里说一下以下的五个类 1.Future2.ConcurrentLinkedQueue3.volatile关键字4.AtomicRefere ...

  5. LVS DR模式 负载均衡服务搭建

    LVS 负载均衡 最近在研究服务器负载均衡,阅读了网上的一些资料,发现主要的软件负载均衡方案有nginx(针对HTTP服务的负载均衡),LVS(针对IP层,MAC层的负载均衡).LVS模式工作在网络层 ...

  6. shiro重新赋值权限

    /** * 重新赋值权限(在比如:给一个角色临时添加一个权限,需要调用此方法刷新权限,否则还是没有刚赋值的权限) * @param myRealm 自定义的realm * @param usernam ...

  7. vCenter Server Appliance

    vCenter Server Appliance https://10.0.0.10:5480

  8. 【xsy1629】可持久化序列 - 可持久化平衡树

    题意 你现在要用数据结构维护一个长度为n的序列. 这个序列支持三种操作: 1 l r:将序列中的第l项到第r项这一段翻转. 2 l r:查询序列中[l,r]这一段的和. 3 p:回到第p个历史版本. ...

  9. DFA敏感词过滤

    import java.io.UnsupportedEncodingException; import java.nio.ByteBuffer; import java.util.ArrayList; ...

  10. &nbsp;空格用法

    记录一下,空格的转义字符分为如下几种:平时一般用的是 1.  &160#;不断行的空白(1个字符宽度)2.  &8194#;半个空白(1个字符宽度)3.  &8195#;一个空 ...