C# 反射研究
概念
反射这东西,对于我这种小白,听起来总是觉得好大上的。
当初理解它费了一点时间,后来看了一句话,突然恍然大悟,“反射就跟B超一样,我们在不剖开人体的情况下想看清楚内部情况,
我们就通过发射超声波,然后根据超声波反射回来的情况,描绘出人体内部的细节”。
比喻得很形象呢呵呵,C#的反射也差不多这样,我们想要探究一个类(或程序集)内部的情况,
但我们又没有权限进入内部查看,于是我们用反射,然后获取该程序集内部的情况,达到使用引用他的目的。
这是比较官方化的定义:反射就是审查元数据并收集关于它的类型信息的能力。元数据(编译以后的最基本数据单元)
就是一大堆的表,当编译程序集或者模块时,编译器会创建一个类定义表,一个字段定义表,和一个方法定义表等。
作用
反射的作用,说白了就是在程序运行时可以动态获取其他程序集(dll)等的内部情况,然后可以动态加载并使用他们。
1、最常见的一个做法,就是插件。.Net程序的插件写法就是:你预留一些接口,然后编写插件的人根据你预留的接口写插件,
并将编译好的dll放到对应文件夹,那么你的程序运行时就会用反射动态加载使用这些插件。
2、还有一个很牛逼的做法就是可以达到修改程序一些行为的目的,例如Unity的Editor,虽然支持编程插件,但一些API并不对外
开放,你可以通过反编译Unity的dll(前提是你反编译得了),然后获取某个功能是用什么API完成的,然后通过反射加载Editor
的dll,并根据上一步获取的API用法,自己来使用这些不公开的API。
参考雨松MONO的博客:http://www.xuanyusong.com/archives/3796
3、另一个用法就是可以通过函数名加载函数拉,比如本人最近在做一个插件(类PlayMaker的可视化FSM插件),里面有个功能类,
这个功能类里有许多的函数,每个函数对应一个功能,然后在程序运行时会将这些函数映射到屏幕上的一个列表里,你点击列表里的
哪个按钮,就执行哪个函数。虽然可以通过为每个函数绑定一个Button实现,但比如函数非常多,这样就忙不过来了,而且达不到可以
动态修改的目的。比如你以后要多添加几个功能函数,那么又要为它们绑定几个按钮。我的想法就是在创建列表的时候扫描这个类,
将这个类的所有函数列出来,然后为他们绑定按钮。反射帮助了我。(大误:刚刚百度了下,才知道通过函数名加载函数没必要用反射,
直接Type.GetMethod就能返回函数数组了,反射性能不高,没必要用)
参考我这篇文章的第六点:http://www.cnblogs.com/jeason1997/p/4802116.html
注:顺便一说,Unity里的SendMessage就是通过反射调用回调函数的,SendMessage("Methon Name");
使用
//获取类型信息 Type type = typeof(MyClass); Response.Write("类型名:" + type.Name); Response.Write("类全名:" + type.FullName); Response.Write("命名空间名:" + type.Namespace); Response.Write("程序集名:" + type.Assembly); Response.Write("模块名:" + type.Module); Response.Write("基类名:" + type.BaseType); Response.Write("是否类:" + type.IsClass); Response.Write("类的公共成员:"); //得到所有公共成员 MemberInfo[] memberInfos = type.GetMembers(); foreach (var item in memberInfos) Response.Write(string.Format("成员类型" + item.MemberType));
//获取当前执行代码的程序集 Assembly assem = Assembly.GetExecutingAssembly(); Response.Write("程序集全名:"+assem.FullName); Response.Write("程序集的版本:"+assem.GetName().Version); Response.Write("程序集初始位置:"+assem.CodeBase); Response.Write("程序集位置:"+assem.Location); Response.Write("程序集入口:"+assem.EntryPoint); Type[] types = assem.GetTypes(); Response.Write("程序集下包含的类型:"); foreach (var item in types) { Response.Write("类:"+item.Name); }
// 动态加载Dll并反射调用程序集中的方法
protected void Page_Load(object sender, EventArgs e) { System.Reflection.Assembly ass = Assembly.LoadFrom(AppDomain.CurrentDomain.BaseDirectory+"bin\\Test.dll"); //加载DLL System.Type t = ass.GetType("Test.MyClass");//获得类型 string name = typeof(MyClass).AssemblyQualifiedName; System.Type t1 = Type.GetType(name); System.Type t2 = typeof(MyClass); object o = System.Activator.CreateInstance(t);//创建实例 System.Reflection.MethodInfo mi = t.GetMethod("Fun_1");//获得方法 mi.Invoke(o, new object[] { this.Page, "alert('测试反射机制')" });//调用方法 System.Reflection.MethodInfo mi1 = t.GetMethod("Fun_2"); mi1.Invoke(t, new object[] { this.Page, "alert('测试反射机制1')" });//调用方法 }
// 获取某个类的所有子类 var allTypes = Assembly.Load("Assembly-CSharp").GetTypes(); var childTypes = allTypes.Where(type => type.IsSubclassOf(typeof(ParentType))).ToList();
动态创建类
(待编辑)
动态编译,参考这篇文章:http://www.cnblogs.com/lichdr/archive/2004/10/20/54569.html
反射与工厂模式
(待编辑)
参考文章:
http://www.cnblogs.com/binfire/archive/2013/01/17/2864887.html