1.特性是什么?
Attribute 用来对类、属性、方法等标注额外的信息,贴一个标签(附着物)
通俗:给 类 或 类成员 贴一个标签,就像航空部为你的行李贴一个标签一样
其中1、2、3、4、5就是元数据,用来描述数据(6)的数据。
2.特性到底是什么?
如上面的
Obsolete ,会不会也是一个如
public 、
static 这样类似的修饰符呢,我们且看看反编译后的中间语言。
意料之外,我们看到了上面的2、3、4、5,而1(特性)怎么跑到里面去了,且是一种看不懂的东东,反正我们知道了不是类似的修饰符。 然后我们接着在vs里面把光标移到
Obsolete 上按F12,如:
原来只是一个继承了
Attrbute 的一个类(class)。那么上面我们看不懂的部分应该就是这个
ObsoleteAttribute 类的实例化了。
我们来回答上面的问题:特性到底是什么?特性只是一个类而已。
我们看到上面系统特性
Obsolete 上面还有特性,如:
Serializable 、
AttributeUsage 、
Camvisible 等。像这种特性我们称之为“元数据的元数据”(元元数据)。
ComVisible :微软定义“控制程序集中个别托管类型、 成员或所有类型对COM的可访问性”。
AttributeUsage :这个比较重要了,基本上每个特性定义都用到了它。它就是用来表示当前这个特性可用于哪些对象。如:类、方法、属性...等等。(只需要用到这个我们就可以自定义特性了)
就是我们特性名明明是
Obsolete ,为什么我们F12进去后变成了
ObsoleteAttribute 呢?这其实只是一个微软的约定而已,没有为什么。
其实我们可以两种写法:
[ObsoleteAttribute("已过时")] 和
[Obsolete("已过时")] 是等效的,只是我们一般都用后面这种。
4.属性没有
set 方法。只能通过构造函数赋值。(这是因为特性语法所致,因为特性的定义只存在单行的中括号中,不能实例化之后在设置属性,所以全部的设置都在后面的小括号里进行的。如果需要有
set 属性,我们就要用到命名参数,下面会继续讲到)
好了,我们通过这四点完全可以自己定义个特性来玩玩了。我们来定义一个给机器看的注释。我们平时的注释都只是给程序员看的,编译之后就全没了。那我们想在代码运行时,弹出我们的注释怎么办,接下来我们用自定义特性来实现,如:
[AttributeUsage(AttributeTargets.All)] public class TMessgAttribute : Attribute { public TMessgAttribute() { } public TMessgAttribute(string createTime, string createCreatename, string mess) { this.createTime = createTime; this.mess = mess; this.createname = createCreatename; } private string createname; private string mess; private string createTime; public string CreateTime { get { return createTime; } } public string Mess { get { return mess; } } public string Createname { get { return createname; } } } |
上面是自定义的特性,怎么使用,和系统特性一样,先定义个测试类,
[TMessg("2016-12-15", "tangsansan", "我,我只是测试自定义特性,不要报错哦")] public class TClass { } |
我们定义了特性,也使用了特性,然我们却不知道怎么看效果。我们想看到效果怎么办。可以使用反射看看
TClass 类的元数据,如:
System.Reflection.MemberInfo info = typeof(TClass);
TMessgAttribute attr = (TMessgAttribute)Attribute.GetCustomAttribute(info, typeof(TMessgAttribute)); Console.WriteLine("类名:{0}", info.Name); Console.WriteLine("创建时间{0}", attr.CreateTime); Console.WriteLine("创建人{0}", attr.Createname); Console.WriteLine("消息{0}", attr.Mess); Console.WriteLine("-------------"); object[] attrs = info.GetCustomAttributes(typeof (TMessgAttribute), false); TMessgAttribute mes = attrs[0] as TMessgAttribute; Console.WriteLine("创建时间{0}", mes.CreateTime); Console.WriteLine("创建人{0}", mes.Createname); Console.WriteLine("消息{0}", mes.Mess); |
4.什么是命名参数?
上面的自定义特性都是通过构造函数设置字段私有字段,然后通过只提供了
get 的属性来访问。那么可否直接在特性里面定义拥有
get 和
set 的属性吗?答案是肯定的。那怎么在使用特性的时候设置这个属性呢?我们接着往下看。
我们接着在自定义特性里面添加一个属性。
public string modifyTime{ get;set;} |
[TMessg("2016-12-15", "tangsansan", "我,我只是测试自定义特性,不要报错哦",modifyTime="2016-11-26")] public class TClass { } |
我们发现,直接在输入了构造函数之后接着设置属性就可以。(这就相当于可选参数了,属性当然可以随便你是否设置了。不过这里需要注意了,前面的参数一定要按照定义的特性构造函数的参数顺序)
这种参数,我们称为命名参数。
6.我们来继续要看看AttributeUsage (这个描述特性的特性--“元元数据”)
我们来看看他的这几个属性是干嘛的。从最后一个开始看。 1. AttributeTargets
我们在上面其实就已经看到并也已经使用了。
我们设置的是可用于所有对象。
AttributeTargets 其实是个枚举,每个值对于一个类型对象。
你可以直接在
AttributeTargets F12进去:
“如果该属性可由派生类和重写成员继承,则为
true ,否则为
false 。 默认值为
true ”
如下,我们设置
Inherited = false 那么继承
TClass 的
T2Class 无法访问到
TC lass 中设置的特性元数据。
如果我们想要这样设置怎么办。在 AttributeUsage 中设置 AllowMultiple = true 如:
3. AllowMultiple (也是一个布尔值) “如果允许指定多个实例,则为
true ;否则为
false 。 默认值为
false 。”
![196558-20161128151356224-834085900.png](https://images2015.cnblogs.com/blog/196558/201611/196558-20161128151356224-834085900.png)
反之,我们设置 Inherited = true (或者不设置任何,因为默认就是 true )打印如下:
如果我们想要这样设置怎么办。在 AttributeUsage 中设置 AllowMultiple = true 如:
1.添加一个继承类 public class T2Class : TClass { } 2.更新 [AttributeUsage(AttributeTargets.All , Inherited=false)] public class TMessgAttribute : Attribute { } 3.输出![196558-20161128151357021-1912167881.png](https://images2015.cnblogs.com/blog/196558/201611/196558-20161128151357021-1912167881.png) |
但是要报错,
打印地方的代码需要修改。因为之前是打印一个特性信息,这里是打印一个特性数组集合的信息。
TMessgAttribute[] attrs = (TMessgAttribute[])Attribute.GetCustomAttributes(info, typeof(TMessgAttribute)); foreach (var item in attrs) { Console.WriteLine("类名:{0}", info.Name); Console.WriteLine("创建时间{0}", item.CreateTime); Console.WriteLine("创建人{0}", item.Createname); Console.WriteLine("消息{0}", item.Mess); Console.WriteLine("-------------"); Console.WriteLine("修改时间{0}", item.modifyTime); } |
![196558-20161128151357849-1347536492.png](https://images2015.cnblogs.com/blog/196558/201611/196558-20161128151357849-1347536492.png)
7.自定义特性可以干什么?
上面我们通过反编译,发现自定义特性实际上就是一个对象调用的最前面加了一段实例化的代码。
那么我们可以做的事可多了,除了像上面一样为对象设置“注释”,我们还可以自定义个特性,给某些方法或是某些类做“操作日志记录”,或者给需要在执行某些方法的时候需要权限,我们可以做个权限认证的特性等等。
农码一生 《一、特性是什么东东》