目录结构:

contents structure [+]

1.程序集

1.1 程序集的加载

想必读者对程序集并不陌生吧,在这里笔者结合反射来阐述程序集的加载。如果想要动态加载程序集,那么就要使用System.Reflection.Assembly类。
程序集加载最常用的静态方法是:

  1. public static Assembly Load(string assemblyString)
  2. public static Assembly LoadFrom(string assemblyFile)

使用Load方法加载程序集必须传递程序集的全名称,程序集的全名称由四个部分组成(程序集的简单名称、程序集的版本、程序集的语言文化,程序集的公钥标记)。程序集的实例属性FullName就是程序集的全名称。LoadFrom方法允许从本地和远程网络加载程序集,在通过远程网络加载的是时候需要注意,不能通过FTP协议远程加载程序集。

如果只想通过反射来分析程序集中的元数据,并希望程序集中的任何代码都不会被执行,那么在加载程序集的时候最好使用Assembly的ReflectionOnlyLoad方法或是ReflectionOnlyLoadFrom方法。如下:

  1. public static Assembly ReflectionOnlyLoad(string assemblyString)
  2. public static Assembly ReflectionOnlyLoadFrom(string assemblyFile)

除了使用Assembly类来加载程序集之外,还可以利用System.AppDomain类的实例方法public Assembly Load(string assemblyString)来完成。该实例方法有许多重载版本,这里需要注意,由于Assembly不是派生自MarshalByRefObject类,所以使用Load方法得到的程序集是按值封送。
如下:
如果代表当前AppDomain的域为A,Load方法调用在AppDomain域B中,那么被Load加载的程序集会被域A和域B都加载。

  1. AppDomain ad = AppDomain.CreateDomain("ChildDomain");
  2. ad.Load("MyAssembly");

这段代码中,MyAssembly程序集将会被加载到两个AppDomain中,一个是运行代码所在的AppDomain,另一个是新创建的AppDomain。这是因为Assembly是按值封送的,所以CLR将会在调用线程的AppDomain中重新加载程序集。

1.2 发现程序集中的类型

在上面的一节中,我们知道了如何加载一个程序集,接下来就是要探索程序集中的类型。
如下:

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. Assembly assembly = Assembly.Load("System.Data,Version=4.0.0.0,Culture=neutral,PublicKeyToken=b77a5c561934e089");
  6. foreach (Type type in assembly.GetExportedTypes())
  7. {
  8. Console.WriteLine(type.FullName);
  9. }
  10. Console.ReadLine();
  11. }
  12. }

这里笔者加载了System.Data程序集,System.Data程序集是被安装到GAC中的。
GetExportedTypes()返回程序集中由public定义的类型,该方法返回和ExportedTypes属性的含义相同。

2.反射对成员的常规操作

2.1 发现类型的成员

字段、构造器、方法、属性、事件和嵌套类型都可以定义为类型成员。FCL包含了抽象基类System.Reflection.MemberInfo,封装了类型成员的通用属性。MemberInfo有许多派生类,这些类型的层次结构如下:

System.Type 类对于反射起着核心的作用。但它是一个抽象的基类,Type有与每种数据类型对应的派生类,我们使用这个派生类的对象的方法、字段、属性来查找有关该类型的所有信息。
获取给定类型的Type引用有3种常用方式:
使用 C# typeof 运算符。

  1. Type t = typeof(string);

使用对象GetType()方法。

  1. string s = "grayworm";
  2. Type t = s.GetType();

还可以调用Type类的静态方法GetType()。

  1. Type t = Type.GetType("System.String");

除了System.Type类还有一个类是在反射中经常需要使用的,就是BindingFlags。BindingFlags是一个枚举类,经常需要用的枚举值为BindingFlags.Public,BindFlags.NonPublic,BindFlags.Instance,BindFlags.Static,关于这些值的详细含义和更多信息可以参考官方文档。

上面这三类代码都是获取string类型的Type,在取出string类型的Type引用t后,我们就可以通过t来探测string类型的结构了。

  1. string n = "grayworm";
  2. Type t = n.GetType();
  3. foreach (MemberInfo mi in t.GetMembers())
  4. {
  5. Console.WriteLine("{0}/t{1}",mi.MemberType,mi.Name);
  6. }

查看类中的构造方法

  1. NewClassw nc = new NewClassw();
  2. Type t = nc.GetType();
  3. ConstructorInfo[] ci = t.GetConstructors(); //获取类的所有构造函数
  4. foreach (ConstructorInfo c in ci) //遍历每一个构造函数
  5. {
  6. ParameterInfo[] ps = c.GetParameters(); //取出每个构造函数的所有参数
  7. foreach (ParameterInfo pi in ps) //遍历并打印所该构造函数的所有参数
  8. {
  9. Console.Write(pi.ParameterType.ToString()+" "+pi.Name+",");
  10. }
  11. Console.WriteLine();
  12. }

用构造函数动态生成对象

  1. Type t = typeof(NewClassw);
  2. Type[] pt = new Type[];
  3. pt[] = typeof(string);
  4. pt[] = typeof(string);
  5. //根据参数类型获取构造函数
  6. ConstructorInfo ci = t.GetConstructor(pt);
  7. //构造Object数组,作为构造函数的输入参数
  8. object[] obj = new object[]{"grayworm","hi.baidu.com/grayworm"};
  9. //调用构造函数生成对象
  10. object o = ci.Invoke(obj);
  11. //调用生成的对象的方法测试是否对象生成成功
  12. //((NewClassw)o).show();

类的属性(Property)

  1. NewClassw nc = new NewClassw();
  2. Type t = nc.GetType();
  3. PropertyInfo[] pis = t.GetProperties(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
  4. foreach(PropertyInfo pi in pis)
  5. {
  6. Console.WriteLine(pi.Name);
  7. }

查看类中的方法(Method)

  1. NewClassw nc = new NewClassw();
  2. Type t = nc.GetType();
  3. MethodInfo[] mis = t.GetMethods(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);//这里获得的只是获得实例的public和非public修饰的方法,要获得所有的方法,参照上面的获取属性的代码,指定标识符就可以了
  4. foreach (MethodInfo mi in mis)
  5. {
  6. Console.WriteLine(mi.ReturnType+" "+mi.Name);
  7. }

查看类的字段 (Field)

  1. NewClassw nc = new NewClassw();
  2. Type t = nc.GetType();
  3. FieldInfo[] fis = t.GetFields(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
  4. foreach (FieldInfo fi in fis)
  5. {
  6. Console.WriteLine(fi.Name);
  7. }

用反射生成对象,并调用属性、方法和字段进行操作

  1. NewClassw nc = new NewClassw();
  2. Type t = nc.GetType();
  3. Type[] pt = new Type[];
  4. pt[] = typeof(string);
  5. pt[] = typeof(string);
  6. //根据参数类型获取构造函数
  7. ConstructorInfo ci = t.GetConstructor(pt);
  8. //构造Object数组,作为构造函数的输入参数
  9. object[] objpara = new object[]{"grayworm","hi.baidu.com/grayworm"};
  10. //调用构造函数生成对象
  11. object obj = ci.Invoke(objpara);
  12. //取得ID字段
  13. FieldInfo fi = t.GetField("ID",BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
  14. //给ID字段赋值
  15. fi.SetValue(obj, "k001");
  16. //取得MyName属性
  17. PropertyInfo pi1 = t.GetProperty("MyName",BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);
  18. //给MyName属性赋值
  19. pi1.SetValue(obj, "grayworm", null);
  20. PropertyInfo pi2 = t.GetProperty("MyInfo");
  21. pi2.SetValue(obj, "hi.baidu.com/grayworm", null);
  22. //取得show方法
  23. MethodInfo mi = t.GetMethod("show");
  24. //调用show方法
  25. mi.Invoke(obj, null);

2.2 创建类型的实例

在上面介绍类型成员的时候,已经使用类型的构造方法创建过类型的实例了。接下来,进行总结一下:
a.System.Activator的CreateInstance方法。
b.System.Activator的CreateInstanceFrom方法。
c.System.AppDomain的方法
AppDomain类型提供了4个用于构造类型实例的实例方法,包括CreateInstance,CreateInstanceAndUnwrap,CreateInstanceFrom和CreateInstanceFromAndUnwrap方法。
CreateInstance提供了几个重载版本,每个重载版本都返回了一个ObjectHandle类对象。通过调用ObjectHandler的Unwrap就可以得到实例对象。而CreateInstanceAndUnwrap方法,简化了这步操作,该方法返回了一个Object对象。
d.System.Reflection.ConstructorInfo的Invoke方法。

上面列出的机制,可以为除了数组(System.Array派生的类型)和委托(System.MulticastDelegate派生的类型)之外的所有的类型创建类型实例。
创建数组对象,需要使用Array的静态CreateInstance方法。创建委托对象,需要使用MethodInfo的静态CreateDelegate方法。

构造泛型类型的实例,和构造普通实例类型的代码差不多,只是复杂一点而已。构造泛型类型实例的时候,需要使用Type类的MakeGenericType方法创建泛型类型参数。
例如:

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. //获取对泛型类型的类型对象的引用
  6. Type openType=typeof(Dictionary<,>);
  7. //使用TKey=String,TValue=Int32封闭泛型类型
  8. Type closedType = openType.MakeGenericType(typeof(String),typeof(Int32));
  9. //构造封闭类型的实例
  10. Object o = Activator.CreateInstance(closedType);
  11. Console.WriteLine(o.GetType());
  12. Console.ReadLine();
  13. }
  14. }

2.3 绑定句柄减少进程的内存消耗

如果在应用程序中需要绑定一组类型(Type对象)或类型成员(MemberInfo派生对象),并将这些对象保存在某种形式的集合中。以后,应用程序搜索这个集合,查找特定对象,然后调用对象。这种机制很好,但是有个小问题:Type对象和MemberInfo对象都需要大量内存。如果应用程序容纳了太多这样的对象,但只是偶尔使用,应用程序消耗的内存就会急剧上升,这样会对应用程序产生反面影响。

CLR内部是更精简的方式表示这种信息。CLR之所以为引用程序创建这些对象,只是为了方便开发人员。如果需要保存/缓存大量Type和MemberInfo派生对象,开发人员可以使用运行时句柄代替对象减少工作集(占用的内存)。FCL定义了三个运行时句柄对象,包括RuntimeTypeHandle,RuntimeFieldHandle和RuntimeMethodHandle。这三个都是轻量级别的句柄类型,因此如果需要大量缓存Type对象和MemberInfo派生对象,可以考虑使用轻量级的句柄对象。

将Type对象转化为RuntimeTypeHandle,调用Type的静态GetTypeHandle方法并且传递那个Type对象的引用。
将RumtimeTypeHandle转化为Type对象,调用Type的静态方法GetTypeFromHandle,并传递那个RuntimeTypeHandle。
将FieldInfo对象转化为RuntimeFieldHandle,调用FieldInfo的只读实例属性FieldHandle。
将RuntimeFieldHandle转化为FieldInfo对象,调用FieldInfo的静态方法GetFieldFromHandle方法。
将MethodInfo对象转化为RuntimeMethodHandle,调用MethodInfo的只读实例属性MethodHandle。
将RuntimeMethodHandle转化为MethodInfo对象,调用MethodInfo的静态方法GetMethodFromHandle方法。
例如:

  1. class Program
  2. {
  3. private const BindingFlags c_bf = BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.Static;
  4.  
  5. static void Main(string[] args)
  6. {
  7. List<MethodBase> methodInfos=new List<MethodBase>();
  8.  
  9. foreach (Type t in typeof(Object).Assembly.GetExportedTypes()) {
  10. if (t.IsGenericTypeDefinition) continue;
  11. methodInfos.AddRange(t.GetMethods(c_bf));
  12. }
  13. //绑定所有方法后,显示方法的个数
  14. Console.WriteLine("# of methods = {0}",methodInfos.Count);
  15. Show("After building cache of MethodInfo objects");
  16.  
  17. List<RuntimeMethodHandle> methodHandles= methodInfos.ConvertAll<RuntimeMethodHandle>(mb => {
  18. return mb.MethodHandle;
  19. });
  20. Show("Holding MethodInfo and RuntimeMethod Catch");
  21. GC.KeepAlive(methodInfos);//阻止缓存被过早回收
  22.  
  23. methodInfos = null;
  24. Show("After free methodInfo object");
  25.  
  26. methodInfos = methodHandles.ConvertAll<MethodBase>(mrh =>
  27. {
  28. return MethodBase.GetMethodFromHandle(mrh);
  29. });
  30. Show("size of heap after re-creating MethodInfo objects");
  31. GC.KeepAlive(methodInfos);//阻止缓存被过早回收
  32. GC.KeepAlive(methodHandles);//阻止缓存被过早回收
  33.  
  34. methodInfos = null;
  35. methodHandles = null;
  36.  
  37. Show("After freeing both methodInfo and methodHandles");
  38.  
  39. Console.ReadLine();
  40. }
  41. static void Show(String s) {
  42. Console.WriteLine("Heap size={0:N12} - {1}",GC.GetTotalMemory(true),s);
  43. }
  44. }

输出如下:
# of methods = 36572
Heap size=3,881,636.000000000000 - After building cache of MethodInfo objects
Heap size=4,028,180.000000000000 - Holding MethodInfo and RuntimeMethod Catch
Heap size=4,028,140.000000000000 - After free methodInfo object
Heap size=3,994,152.000000000000 - size of heap after re-creating MethodInfo objects
Heap size=153,128.000000000000 - After freeing both methodInfo and methodHandles

3.解析自定义特性

为了简便编程,我们往往需要自定义特性,在程序中可以通过反射获得特性的相关信息。
例如:

  1. class Program
  2. {
  3. [MyTag("test")]
  4. public void init() {
  5. }
  6. static void Main(string[] args)
  7. {
  8. Type cp= typeof(Program);
  9. //获得指定的方法
  10. MethodInfo method= cp.GetMethod("init",System.Reflection.BindingFlags.Public|System.Reflection.BindingFlags.NonPublic|System.Reflection.BindingFlags.Instance);
  11. //获得特性
  12. MyTag attr = method.GetCustomAttribute(typeof(MyTag)) as MyTag;
  13. Console.WriteLine(attr.value);
  14. Console.ReadLine();
  15. }
  16. }
  17. [AttributeUsage(AttributeTargets.Method, Inherited = true)]
  18. class MyTag : System.Attribute
  19. {
  20. public string value = String.Empty;
  21. public MyTag(String info)
  22. {
  23. this.value = info;
  24. }
  25. }

【C#】解析C#程序集的加载和反射的更多相关文章

  1. .net 程序集的加载与反射

    一. 程序集的加载: 在CLR内部使用System.Reflection.Assembly类的静态LoadFrom方法尝试加载程序集. LoadFrom方法在内部调用Assembly的Load方法,将 ...

  2. Clr Via C#读书笔记---程序集的加载和反射

    #1 加载程序集 Assembly.Load: public class Assembly { public static Assembly Load(AssemblyName assemblyRef ...

  3. 深入解析 composer 的自动加载原理 (转)

    深入解析 composer 的自动加载原理 转自:https://segmentfault.com/a/1190000014948542 前言 PHP 自5.3的版本之后,已经重焕新生,命名空间.性状 ...

  4. .net加载失败的程序集重新加载

    在.net程序中,程序集是Lazy加载的,只有在用的时候才会去加载,当程序集加载失败时,会触发AppDomain.AssemblyResolve的事件,在这个事件中,我们甚至还可以进行补救,从别得地方 ...

  5. 显示名为“xxx.XmlSerializers”的程序集未能加载到 ID 为 1 的 AppDomain 的“LoadFrom”绑定上下文中。

    VS调试程序运行中提示“显示名为“xxx.XmlSerializers”的程序集未能加载到 ID 为 1 的 AppDomain 的“LoadFrom”绑定上下文中.错误的原因为: System.IO ...

  6. clr via c# 程序集加载和反射(2)

    查看,clr via c# 程序集加载和反射(1) 8,发现类型的成员: 字段,构造器,方法,属性,事件,嵌套类型都可以作为类型成员.其包含在抽象类MemberInfo中,封装了所有类型都有的一组属性 ...

  7. clr via c# 程序集加载和反射集(一)

    1,程序集加载---弱的程序集可以加载强签名的程序集,但是不可相反.否则引用会报错!(但是,反射是没问题的) //获取当前类的Assembly Assembly.GetEntryAssembly() ...

  8. 【C#进阶系列】23 程序集加载和反射

    程序集加载 程序集加载,CLR使用System.Reflection.Assembly.Load静态方法,当然这个方法我们自己也可以显式调用. 还有一个Assembly.LoadFrom方法加载指定路 ...

  9. .NET:强签名程序集的加载问题 之 版本重定向

    背景 多数解决方案会包含多个项目,某些支持插件架构的解决方案中,更是包含多个插件项目,这些项目会使用一些第三方NuGet Packages,如果管理不慎,解决方案中会出现多个版本的引用,这在编译期间不 ...

随机推荐

  1. NDK环境搭建方法2

    1.新建项目NDKDemo3 2.新建com.example.shixm.ndkdemo3.MyNdk.java 3.右键main文件夹,New->Folder->JNI Folder 4 ...

  2. DataGridView中DataGridViewComboBoxColumn的一些相关应用(一)让其值改变时触发事件-转

    转自  https://maodaili.de/mao.php?u=a%2FMrbEvUE8PnCuc7FrhJi0Rqd3kmOBHPZUbcJ1c2hbJUK0RYWpAf4lhIOddItP%2 ...

  3. poj1256(贪心+并查集)

    题目链接:http://poj.org/problem?id=1456 题意:给n件商品的价格和卖出截至时间,每一个单位时间最多只能卖出一件商品,求能获得的最大利润. 思路:首先是贪心,为获得最大利润 ...

  4. Websocket实现群聊、单聊

    Websocket 使用的第三方模块:gevent-websocket 群聊 ws群聊.py中的内容 from flask import Flask, request, render_template ...

  5. AWVS结果分析与实践-XSS

      今天趁着老师接项目,做了一丢丢实践,以下是一点点感触.     都知道AWVS是神器,可是到我手里就是不灵.拿了它扫了一个URL,结果提示XSS漏洞,实践没反应,只好愉快地享受了过程.来看看.   ...

  6. Codeforces Round #518 (Div. 2) [Thanks, Mail.Ru!]

    Codeforces Round #518 (Div. 2) [Thanks, Mail.Ru!] https://codeforces.com/contest/1068 A #include< ...

  7. Kylin 与 Spark SQL相比,有哪些差异和优势

    SparkSQL本质上是基于DAG模型的MPP.而Kylin核心是Cube(多维立方体).关于MPP和Cube预处理的差异,重复如下: > MPP [1] 的基本思路是增加机器来并行计算,从而提 ...

  8. multithreading coding

    分类:公共资源问题.公交车问题 顺序:Qseater lab, bank, doctor [饭店] geust //yuec2 Yue Cheng package lab9; public abstr ...

  9. 15-算法训练 P1103

    http://lx.lanqiao.cn/problem.page?gpid=T372   算法训练 P1103   时间限制:1.0s   内存限制:256.0MB      编程实现两个复数的运算 ...

  10. 一个命令查看mysql的所有配置(原创)

    在mysql的命令提示符下,执行下面一句话,查看mysql服务器的所有全局配置信息: mysql> show global variables; 得到: 上表的文本内容: "Varia ...