首页 > 代码库 > 《Inside C#》笔记(十四) 反射

《Inside C#》笔记(十四) 反射

通过反射可以在运行时动态地获取一个应用的元数据。

 

一 反射相关的类和方法

与反射相关的类处在System.Reflection命名空间下,包括Assembly、Module、MethodInfo、FieldInfo、PropertyInfo、EventInfo。

a)Type

System.Type类是反射机制的核心。Type类本身是一个抽象类,代表CTS通用类型系统中的某个类型。

对于已经实例化的对象,可以使用<对象名称>.GetType()来取得Type类。另外也可以使用静态方法Type.GetType(“<type name>”)取得未实例化对象的type类。使用第二种方法时,<type name>必须是CTS类型系统的类型名称,而不是C#别名,这可以确保.NET平台下的多语言通用性。

b)使用Type类

通过Type类,可以使用Type.IsByRef\IsClass\IsEnum\IsExplicitLayout\IsInterface\IsPublic\IsSealed等各种各样的方法来取得所需的信息。

 

二 通过反射使用程序集或模块

程序集是一个包含了若干PE(Portable Executable)文件的物理文件,.NET中对应的类为Assembly。通过Assembly类,可以进行如下操作:

遍历程序集内部的所有类型

取得程序集的所有模块

获取程序集的文件名、物理路径

查看程序集的版本及安全信息

取得程序集的入口

通过方法Assebmbly a= Assembly.LoadFrom(<程序集路径>)可以加载指定的程序集文件,然后Type[] types=a.GetTypes()可以取得程序集中的所有类型组成的数组。

 

三 使用反射实现晚绑定

有些应用场景下,在编译时无法确定所有事情(比如包含哪些DLL),例如一个软件的插件,这时需要使用晚绑定技巧,让软件在运行时动态的决定要加载的DLL或程序集。

具体的方式为

 

技术分享

 

 

在加载程序集文件并使用GetType()取得所有的类型后,通过使用Type.Is***系列方法做判断,找到所需的type,然后使用Activator.CreateInstance()激活该类型,随后就可以取得MethodInfo并调用Invoke来执行了。

四在运行时创建并执行代码

借助System.Reflection.Emit命名空间的内容,可以在内存动态地定义程序集、模块、类型,并将生成的代码插入程序中执行。

例如,有生成HelloWorld方法的代码:

 

技术分享

技术分享

?

 

 

看起来步骤挺多的,但遵循程序集-模块-类型-方法-方法体这样的创建顺序。然后就可以按照先前的晚绑定方法来使用了。

 

学习资料:Inside C# by Tom Archer

 

《Inside C#》笔记(十四) 反射