使用特性,可以有效地将元数据或声明性信息与代码(程序集、类型、方法、属性等)相关联。 将特性与程序实体相关联后,可以在运行时使用反射这项技术查询特性。
https://learn.microsoft.com/zh-cn/dotnet/csharp/programming-guide/concepts/attributes/
上面是引自微软官方文档,关于特性的作用的描述。在.NET7中,可以给特性定义泛型了。这看起来似乎和特性的参数有矛盾,因为特性的参数是以下类型。.
-
简单类型:bool,byte,char,double,float,int,long,sbyte,short,string,uint,ulong,ushort。
-
object类型。
-
System.Type类型。
-
枚举类型。
-
上面类型的一维数组。
看下面的例子,定义泛型参特性,想把T当构造函数参数,也就是特性的必值参数使用,但是这里报错了,信息是:这不是有效的特性参数类型,这也就是上面说的特性参数与泛型有矛盾的地方。
/// <summary>/// 自定义类/// </summary>public class MyType{}/// <summary>/// 泛型特性/// </summary>/// <typeparam name="T"></typeparam>public class GenericAttribute<T> : Attribute where T : class, new(){public GenericAttribute(T t){}}/// <summary>/// 使用泛型特性/// </summary>[Generic<MyType>(new MyType())]public class A{}
其实泛型特性主要是提供声明性信息的,不能传A类型的实例对像是可以理解的,至少能把A类型这个声明性信息带入。所以这就对泛特性的使用姿势有要求了。
还是先看一下面的一个用法吧。
using System.Reflection;using System.Text;var person = new Person();person.ID = 10;person.Name = "桂素伟";Console.WriteLine(person);var order = new Order();order.ID = 10;order.Name = "批发";order.Pirce = 12.34m;Console.WriteLine(order);[AttributeUsage(AttributeTargets.Class, AllowMultiple = false)]public class DataFormatAttribute<T> : Attribute{}public interface IFormatter{string ConvertTo(object obj);}public class JsonFormatter : IFormatter{public string ConvertTo(object obj){return System.Text.Json.JsonSerializer.Serialize(obj);}}public class XmlFormatter : IFormatter{public string ConvertTo(object obj){var ser = new System.Xml.Serialization.XmlSerializer(obj.GetType());using var memory = new MemoryStream();ser.Serialize(memory, obj);var bytes = memory.GetBuffer();return Encoding.UTF8.GetString(bytes, 0, bytes.Length);}}public abstract class Entity{public override string? ToString(){var type = this.GetType();foreach (var attr in type.GetCustomAttributes(false)){if (attr.GetType().IsGenericType){var pars = attr.GetType().GenericTypeArguments;foreach (var par in pars){var format = Activator.CreateInstance(par) as IFormatter;return format?.ConvertTo(this);}}}return null;}}[DataFormat<JsonFormatter>()]public class Person : Entity{public int ID { get; set; }public string? Name { get; set; }}[DataFormat<XmlFormatter>()]public class Order : Entity{public int ID { get; set; }public string? Name { get; set; }public decimal Pirce { get; set; }}
例子很简单,就是把实体类的格式化器声明在实体定义阶段,最终通过重构ToString函数,实现对泛特性的类型参数反射和使用。由此例可以大体看出,泛型特性的T,一般是通过反射创建实例而来的,而不是通过定义实例化而来的,所以这个类型一般是使用其中的方法等功能,而非属性类的数据。