对于用我们所熟悉的C#源代码编写的类,通过查看方法头或相关类定义中的变量声明,我们很容易确定其类型。当我们的源代码成为一个被外界使用的组件,判断类型属性仍很重要(也许我们的组件为一个数据库提供数据或与组件交换数据,且此组件用于C#语法和配置不同的计算机语言编写而成,甚至我们的组件要作为WEB服务在因特网上传送)。
为使我们的组件能以一种灵活、用户友好的方式描述类型,从而与不同的应用环境无缝集成,我们使用到了C#中内置的元数据(编译器自动将相关元数据发送到一个程序的功能),元数据包含了方法、实例变量和程序中每个类型许多其他重要性质的详细描述(类型的名字),而在运行时根据程序集及其中的类型得到元数据的过程就叫反射(Reflection)。
你首先要理解一下概念:
一 类型(Type) 对象是什么
比如 object x; x是对象,object就是它的类型,在程序中如何描述类型这个概念呢?就是Type(System.Type)。要获取某个类的类型可以用typeof()操作符
object a;object b;
DataTable t;
Type aType = typeof(object);Type bType = typeof(object);tType = typeof(DataTable);
aType==bType!=tType;
二 程序集(Assembly)
就是你IDE生成的.exe或.dll文件的运行时就叫程序集。所有的代码都在程序集中。你可以通过Assembly.Load()系列函数动态加载程序集(这一步是动态+载代码的前提,因为所有的代码都在程序集中)。
三 动态加载
我们普通调用代码是: 对象名.方法名(参数列表);
class a{
void func(int x){}
public static void Main(string[] args)
{
//创建对象
a a1 = new a();
//调用函数
a1.func(1);
}
}
用反射动态调用代码是
//加载程序集
System.Reflection.Assembly asm = Assembly.LoadFile(assemblyPath);
//获取类型
Type aType = asm.GetType("名字空间.类名");
//获取没有参数的构造函数
System.Reflection.ConstructorInfo conn = t.GetConstructor(new Type[0]);
//调用没有参数的构造函数,Invoke返回object它其是a类
object a1 = conn.Invoke(new object[0]);
//获取参数类型为int,函数名为func的方法
MethodInfo method = t.GetMethod("func",new Type[]{typeof(int)});
//在a1上调用func方法,参数为1
method.Invoke(a1,new object[]{1});
动态调用(后一种方法)比静态调用更复杂,而且效率大概低20倍(网上有个哥们好像测试过)。只有在特殊的时候才调用动态加载动态调用---比如,你的主程序启动的时候子模块还没有,要根据登陆信息下载子模块代码并调用子模块代码,就只能用动态+载来实现了.
四 元数据
.net生成的IL代码中标明了在这个(exe,dll)文件中所有的class(类) method(方法)Attribute(属性) Property(类属性)的签名和调用方法信息,这些信息就叫做元数据。所谓的Reflection反射,就是利用元数据,可以了解到某个assembly(基本等同文件)中的class。。。。。 (就是上面那一串咚咚)信息和调用方法。
五 .net framework
在.net framework中反射类基本都在System.Reflection中。System.Type是反射的核心类.
与它相关的还有System.Attribute命名空间。
简单的说,就是从一个对象的Type类型中找到你所定义这个类的一切信息,如类的方法、属性等
我有多个类,用来存贮数据,每个类有多个属性,其中有些属性是必需的,而有些是可选的。每个类,我都需要检查一下,必需的属性是否为空值。
于是我的做法是把这些类全继承自一个DataBase基类,而这个基类实现一个方法,通过反射查询类所有的属性,先判断属性是否有一个自定义的NotNullAttribute,如果有则通过反射查询这个属性的值是否为空,为空则返回false
这样,我的类只要继承自DataBase类,同时在不为空的属性上面加上[NotNull]就可以了
这样子还是很方便的,我不用为每个类去写一个验证函数了