1.特性是什么?

 Attribute 用来对类、属性、方法等标注额外的信息,贴一个标签(附着物) 
通俗:给 类 或 类成员 贴一个标签,就像航空部为你的行李贴一个标签一样 
个人理解,特性就是修饰对象元数据的修饰符。
1.是特性             2.是访问修饰符 
3.声明修饰符     4.数据类型 
5.变量名             6.变量数据值
其中1、2、3、4、5就是元数据,用来描述数据(6)的数据。

2.特性到底是什么?

如上面的 Obsolete  ,会不会也是一个如 public 、 static 这样类似的修饰符呢,我们且看看反编译后的中间语言。
 意料之外,我们看到了上面的2、3、4、5,而1(特性)怎么跑到里面去了,且是一种看不懂的东东,反正我们知道了不是类似的修饰符。
然后我们接着在vs里面把光标移到 Obsolete  上按F12,如:

 原来只是一个继承了 Attrbute 的一个类(class)。那么上面我们看不懂的部分应该就是这个 ObsoleteAttribute 类的实例化了。

我们来回答上面的问题:特性到底是什么?特性只是一个类而已。


3.自定义特性

我们看到上面系统特性 Obsolete 上面还有特性,如: Serializable 、 AttributeUsage 、 Camvisible 等。像这种特性我们称之为“元数据的元数据”(元元数据)。
1.我们分别来解释性上面的三个特性。
 Serializable :表示类型支持序列化。
 ComVisible :微软定义“控制程序集中个别托管类型、 成员或所有类型对COM的可访问性”。
 AttributeUsage :这个比较重要了,基本上每个特性定义都用到了它。它就是用来表示当前这个特性可用于哪些对象。如:类、方法、属性...等等。(只需要用到这个我们就可以自定义特性了)
2.上面有个问题,不知道大家发现没有。
就是我们特性名明明是 Obsolete  ,为什么我们F12进去后变成了 ObsoleteAttribute 呢?这其实只是一个微软的约定而已,没有为什么。
其实我们可以两种写法: [ObsoleteAttribute("已过时")]  和  [Obsolete("已过时")]  是等效的,只是我们一般都用后面这种。
3.定义的特性必须继承于 Attribute 。
4.属性没有 set 方法。只能通过构造函数赋值。(这是因为特性语法所致,因为特性的定义只存在单行的中括号中,不能实例化之后在设置属性,所以全部的设置都在后面的小括号里进行的。如果需要有 set 属性,我们就要用到命名参数,下面会继续讲到)
好了,我们通过这四点完全可以自己定义个特性来玩玩了。我们来定义一个给机器看的注释。我们平时的注释都只是给程序员看的,编译之后就全没了。那我们想在代码运行时,弹出我们的注释怎么办,接下来我们用自定义特性来实现,如:

  1. [AttributeUsage(AttributeTargets.All)] //设置可以用于那些对象

  2. public class TMessgAttribute : Attribute

  3. {

  4. public TMessgAttribute()  {    }

  5. public TMessgAttribute(string createTime, string createCreatename, string mess)

  6. {

  7. this.createTime = createTime;

  8. this.mess = mess;

  9. this.createname = createCreatename;

  10. }

  11. private string createname;

  12. private string mess;

  13. private string createTime;

  14. public string CreateTime

  15. {

  16. get { return createTime; }

  17. }

  18. public string Mess

  19. {

  20. get { return mess; }

  21. }

  22. public string Createname

  23. {

  24. get { return createname; }

  25. }

  26. }

上面是自定义的特性,怎么使用,和系统特性一样,先定义个测试类,
  1. [TMessg("2016-12-15", "tangsansan", "我,我只是测试自定义特性,不要报错哦")]

  2. public class TClass

  3. {

  4. }

我们定义了特性,也使用了特性,然我们却不知道怎么看效果。我们想看到效果怎么办。可以使用反射看看 TClass 类的元数据,如:

  1. System.Reflection.MemberInfo info = typeof(TClass);

  2. /*

  3. * 方法一

  4. * Attribute.GetCustomAttribute 方法 (MemberInfo, Type)

  5. * 检索应用于类型成员的自定义参数。参数指定成员和要搜索自定义成员的类型

  6. *        element:一个从 MemberInfo 类派生的对象,该类描述类的构造函数、事件、字段、方法或属性成员。(指定应用了特性的类)。

  7. *  attributeType:要搜索的自定义属性的类型或基类型

  8. */

  9. TMessgAttribute attr = (TMessgAttribute)Attribute.GetCustomAttribute(info, typeof(TMessgAttribute));

  10. Console.WriteLine("类名:{0}", info.Name);

  11. Console.WriteLine("创建时间{0}", attr.CreateTime);

  12. Console.WriteLine("创建人{0}", attr.Createname);

  13. Console.WriteLine("消息{0}", attr.Mess);

  14. Console.WriteLine("-------------");

  15. /*

  16. * 方法二

  17. * MemberInfo.GetCustomAttributes 方法 (Type, Boolean)

  18. * attributeType:要搜索的属性的类型。 仅返回可分配给此类型的属性。

  19. *       inherit:true 搜索此成员继承链,以查找这些属性;否则为 false。 属性和事件,则忽略此参数。

  20. */

  21. object[] attrs = info.GetCustomAttributes(typeof (TMessgAttribute), false);

  22. TMessgAttribute mes = attrs[0] as TMessgAttribute;

  23. Console.WriteLine("创建时间{0}", mes.CreateTime);

  24. Console.WriteLine("创建人{0}", mes.Createname);

  25. Console.WriteLine("消息{0}", mes.Mess);

4.什么是命名参数?

上面的自定义特性都是通过构造函数设置字段私有字段,然后通过只提供了 get 的属性来访问。那么可否直接在特性里面定义拥有 get 和 set 的属性吗?答案是肯定的。那怎么在使用特性的时候设置这个属性呢?我们接着往下看。
我们接着在自定义特性里面添加一个属性。
  1. //修改时间

  2. public string modifyTime{get;set;}

使用自定义特性。
  1. [TMessg("2016-12-15", "tangsansan", "我,我只是测试自定义特性,不要报错哦",modifyTime="2016-11-26")]

  2. public class TClass

  3. {  }

我们发现,直接在输入了构造函数之后接着设置属性就可以。(这就相当于可选参数了,属性当然可以随便你是否设置了。不过这里需要注意了,前面的参数一定要按照定义的特性构造函数的参数顺序)

这种参数,我们称为命名参数。

6.我们来继续要看看AttributeUsage    (这个描述特性的特性--“元元数据”)


我们来看看他的这几个属性是干嘛的。从最后一个开始看。

1. AttributeTargets 

我们在上面其实就已经看到并也已经使用了。


我们设置的是可用于所有对象。 AttributeTargets 其实是个枚举,每个值对于一个类型对象。
你可以直接在 AttributeTargets F12进去:

 我们看到了每个值代表可以用于所对于的对象类型。

2. Inherited (是一个布尔值)
“如果该属性可由派生类和重写成员继承,则为 true ,否则为 false 。 默认值为 true ”
如下,我们设置 Inherited = false 那么继承 TClass 的 T2Class 无法访问到 TClass 中设置的特性元数据。

如果我们想要这样设置怎么办。在 AttributeUsage 中设置 AllowMultiple = true 如:


3. AllowMultiple (也是一个布尔值)
“如果允许指定多个实例,则为 true ;否则为  false 。 默认值为 false 。”
我们设置两个特性试试,如:

反之,我们设置 Inherited = true (或者不设置任何,因为默认就是 true )打印如下:

如果我们想要这样设置怎么办。在 AttributeUsage 中设置 AllowMultiple = true 如:

1.添加一个继承类
  1. public class T2Class : TClass

  2. {

  3. //...........

  4. }

2.更新

  1. [AttributeUsage(AttributeTargets.All , Inherited=false)] //设置可以用于那些对象

  2. public class TMessgAttribute : Attribute

  3. {

  4. //...

  5. }

3.输出


但是要报错,


打印地方的代码需要修改。因为之前是打印一个特性信息,这里是打印一个特性数组集合的信息。

  1. TMessgAttribute[] attrs = (TMessgAttribute[])Attribute.GetCustomAttributes(info, typeof(TMessgAttribute));

  2. foreach (var item in attrs)

  3. {

  4. Console.WriteLine("类名:{0}", info.Name);

  5. Console.WriteLine("创建时间{0}", item.CreateTime);

  6. Console.WriteLine("创建人{0}", item.Createname);

  7. Console.WriteLine("消息{0}", item.Mess);

  8. Console.WriteLine("-------------");

  9. Console.WriteLine("修改时间{0}", item.modifyTime);

  10. }

7.自定义特性可以干什么?

上面我们通过反编译,发现自定义特性实际上就是一个对象调用的最前面加了一段实例化的代码。
 
那么我们可以做的事可多了,除了像上面一样为对象设置“注释”,我们还可以自定义个特性,给某些方法或是某些类做“操作日志记录”,或者给需要在执行某些方法的时候需要权限,我们可以做个权限认证的特性等等。
这里就需要大家自己去扩展了。

农码一生 《一、特性是什么东东》

http://www.cnblogs.com/zhaopei/p/attribute_is_what.html


C#回顾 –6.特性的更多相关文章

  1. html+css学习笔记 3[浮动]

    inline-block/float(浮动) 回顾:inline-block 特性:      1.块在一排显示 2.内联支持宽高 3.默认内容撑开宽度 4.标签之间的换行间隙被解析(问题) 5.ie ...

  2. inline-block的升级float:浮动

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

  3. TensorRT 3:更快的TensorFlow推理和Volta支持

    TensorRT 3:更快的TensorFlow推理和Volta支持 TensorRT 3: Faster TensorFlow Inference and Volta Support 英伟达Tens ...

  4. [C#] C# 知识回顾 - 特性 Attribute

    C# 知识回顾 - 特性 Attribute [博主]反骨仔 [原文地址]http://www.cnblogs.com/liqingwen/p/5911289.html 目录 特性简介 使用特性 特性 ...

  5. QT5.9 新特性与版本回顾

    原文链接: http://blog.qt.io/blog/2017/05/31/qt-5-9-released 翻译内容如下,采用的是第三方某在线翻译软件,所以有些地方不是太精确,纵然大吉做了一定的调 ...

  6. 第十四章 JDK新特性回顾

    14.1.JDK5新特性回顾 自动装箱.拆箱 静态导入 增强for循环 可变参数 枚举 泛型 元数据 14.2.JDK7新特性回顾 对Java集合(Collections)的增强支持 在switch中 ...

  7. Android Lollipop 5.0 经典新特性回顾

    *Tamic 专注移动开发! 更多文章请关注 http://blog.csdn.net/sk719887916 虽然Android已到了7.0 ,但是我们还是不能忘怀视觉革命性改变的5.0,今天回顾下 ...

  8. Java9发布回顾Java 8的十大新特性

    java9已经在北京时间9月22日正式发布,开发者可以在oracle jdk官网上下载到最新的jdk9. 今天,我们先来一起复习一下2014年发布的Java 8的十大新特性.先来喝杯java~~~ 按 ...

  9. Java知识回顾 (18)Java 8、9、11的新特性

    Java 8 Java 8 (又称为 jdk 1.8) 是 Java 语言开发的一个主要版本. Oracle 公司于 2014 年 3 月 18 日发布 Java 8 ,它支持函数式编程,新的 Jav ...

随机推荐

  1. 7.Android开源项目WheelView的时间和地址联动选择对话框

    类似WheelView的时间和地址联动选择对话框在现在App经常看到,今天小结下. 主布局界面: <LinearLayout xmlns:android="http://schemas ...

  2. AOJ DSL_2_E Range Add Query (RAQ)

    Range Add Query 数列 A = {a1,a2,...,an} に対し.次の2つの操作を行うプログラムを作成せよ. add(s,t,x): as,as+1,...,at にxを加算する. ...

  3. Java代码执行顺序(静态变量,非静态变量,静态代码块,代码块,构造函数)加载顺序

    //据说这是一道阿里巴巴面试题,先以这道题为例分析下 public class Text { public static int k = 0; public static Text t1 = new ...

  4. 入手了[云梯的VPN]--水文

    之前写的文章 http://www.cnblogs.com/rollenholt/p/3783084.html 结果很多朋友都说访问不了了,现在重新发一下: 各位看官,这是一篇水文: 在用了一段时间s ...

  5. easyUi datagrid 返回时间格式化操作

    1.格式化返回的时间 { field : 'endTime', title : '轮播图片循环的结束时间', width : 50, align : 'center' //格式化时间操作 format ...

  6. WebUploader UEditor chrome 点击上传文件选择框会延迟几秒才会显示 反应很慢

    chrome52.0.2743.80以上, accept: { title: 'Images', extensions: 'jpg,jpeg,png', mimeTypes: 'image/*' } ...

  7. 如何同时打开两个excel

    1. 打开一个excel1 2. 不要双击想要打开的excel2.右键excel应用的图标,选择excel2007. 3. 将excel2拖动到2所打开的新建excel中. 4. over.

  8. SaaS模式的软件

     SaaS是Software-as-a-Service(软件即服务)的简称,随着互联网技术的发展和应用软件的成熟, 在21世纪开始兴起的一种完全创新的软件应用模式.它与"on-demand  ...

  9. JS策略模式

    1.策略模式的定义 策略模式又叫算法簇模式,将一组算法分装到一组具体共同接口的独立类或者对象中,它不影响客户端的情况下发生变化. 通常策略模式适用于当一个应用程序需要实现一种特点的服务和功能,而且该程 ...

  10. C#成员函数直接调用和反射+委托的性能比较

    using System; using System.Reflection; using System.Diagnostics; namespace Refl { class Test { publi ...