反射是动态操作元数据的能力,从程序集-模块-类型-字段-属性-方法-参数等,反射可以动态操作这些部分,在编译时不确定,在运行中动态确定,并进行有效控制。
今天本文主要聊聊Type类,反射的核心是Type类,这个类封装了关于类型的元数据,也是进行反射的入口。
当获得了类型的Type对象后,就可以根据Type提供的属性和方法获取这个类型的一切信息,包括构造函数、字段、属性、事件、方法、参数等。
获取已加载程序集中类型的Type对象,一般来说有以下三种方法:
1.使用Type类提供的静态方法GetType()
GetType方法接受字符串形式的类型名称
Type t = Type.GetType("System.IO.Stream");
2.使用typeof操作符
ypeof操作符来完成这一操作,有点像泛型,Stream就好像一个类型参数一样,传递到typeof操作符中。
Type t = typeof(System.IO.Stream);
3.通过类型实例获得Type对象
在类型实例上调用继承自System.Object的GetType()方法来获得Type对象,使用这种方法时应当注意,返回的Type对象本身不包含关于特定对象的特定信息。
Stream memoryStream = new MemoryStream();
memoryStream.WriteByte(1);
Type t = memoryStream.GetType();
这里的t并不知道memoryStream实例对象里面有哪些信息。
Type类型及System.Reflection命名空间的组织结构
假设我们顺利的拿到了Type对象t:
MemoryStream memoryStream = new MemoryStream();
memoryStream.WriteByte(1);
Type t = memoryStream.GetType();
现在通过t,我们可以获得哪些信息?
1. memoryStream对象的类型的基本信息
通过上图,可以获得:
memoryStream对象的类型,
类型的命名空间,
类型的基类,
在.NET运行库中的映射类型,
是public、private,还是protected,
是值类型还是引用类型,
是不是基元类型,
是枚举、类、数组,还是接口,
2. memoryStream对象的类型的成员信息
除了类型本身的信息以外,还要进一步知道它的成员信息:
a. 包含哪些字段,字段名称、类型、可访问性。
b. 包含哪些属性,属性名称、类型、可访问性。
c. 包含哪些构造函数,构造函数的名称,构造函数的参数个数、参数类型、参数名称。
d. 包含哪些方法,方法的名称,方法的返回值类型,方法的参数个数、参数类型、参数名称。
e. 包含哪些事件,事件的名称。
f. 实现了哪些接口。
g. 可以逐级深入的,比如可以再次获得memoryStream对象的类型的属性的类型的Type信息,然后再次深入下去。
观察上面的列举,先就第一条来说,获取类型都有哪些字段,以及这些字段的信息。字段的信息包括:字段的类型、字段的名称、字段的可访问性(public、private)、字段是否为const、字段是否为readonly,等等,所有这些关于字段的信息也应该抽象为一个类型。
如同上一节所讲到的,System.Reflection命名空间下提供了FieldInfo类型,它封装了关于字段的所有信息。
一样的思路,FCL还给属性,构造器,方法,事件,参数定义了XXXInfo类型:
TypeInfo类型,封装了类的信息;
PropertyInfo类型,封装了类型的属性信息;
ConstructorInfo类型,封装了类型的构造函数信息;
MethodInfo类型,封装了类型的方法信息;
EventInfo类型,封装了类型的事件信息;
ParameterInfo类型,封装了方法和构造函数的参数信息;
最后,注意到Type类型,以及所有以Info结尾的类型均继承自System.Reflection.MemberInfo类型,MemberInfo类型提供了获取类型基础信息的能力。
浏览Type类型的成员,会发现它的属性和方法可以大致分为以下几组:
a. IsXXXX,比如IsAbstract、IsClass,这组属性返回bool类型,用于说明类型是否具备某些特性。
b. GetXXXX(),比如GetField(),返回FieldInfo,这组方法用于获取某一特定成员的信息。
c. GetXXXXs(),比如GetFields(),返回FieldInfo[],这组方法用户获取某些成员信息。
d. 其他的一些属性和方法
由于MemberInfo是一个抽象基类,在获得一个MemberInfo对象后,并不知道它的实际类型是PropertyInfo还是FieldInfo。因此,需要提供方法进行判断,在System.Reflection命名空间中,会遇到很多的位标记,这里先介绍一个位标记(用[Flags]特性标记的枚举,称为位标记),MemberTypes,它用于标记成员类型,取值如下:
[Flags]
public enum MemberTypes {
Constructor = 1, // 该成员是一个构造函数
Event = 2, // 该成员是一个事件
Field = 4, // 该成员是一个字段
Method = 8, // 该成员是一个方法
Property = 16, // 该成员是一个属性
TypeInfo = 32, // 该成员是一种类型
Custom = 64, // 自定义成员类型
NestedType = 128, // 该成员是一个嵌套类型
All = 191, // 指定所有成员类型。
}
本文回顾:
获取某类型的Type对象
某类型的基本信息
某类型的成员信息
MemberInfo类型
Type类型
TypeInfo类型
PropertyInfo类型
ConstructorInfo类型
MethodInfo类型
EventInfo类型
ParameterInfo类型
MemberTypes枚举