反射是.NET很强大的一个机制。

它就像照妖镜一般的存在。它能调用你的任意私有成员,如:私有构造函数、私有方法、私有字段。

类的构造函数声明为了private,别人无法实例化对象出来?No,No,No!!!

下面展示一下反射的魅力:

  1  using System;  2 using System.Reflection;  3 using System.Runtime.Remoting;  4   5 namespace zuo_TestReflectionProjectI{  6   7     #region "程序入口"  8     public class Program{  9         static void Main(){ 10             Console.WriteLine("先载入一个程序集,然后分析它的结构:"); 11             Assembly ass = Assembly.GetExecutingAssembly();    //加载当前程序集 12             Console.WriteLine("程序集的名称是:{0}",ass.FullName); 13             Console.WriteLine(new string('-',40)); 14              15             //列出包含的类 16             Type[] tss = ass.GetTypes(); 17             foreach(Type ts in tss){ 18                 Console.WriteLine("类型名:{0}",ts.Name); 19             } 20              21             Console.WriteLine(new string('-',40)); 22             //单独针对TestClass反射调用 23             Type Oa = ass.GetType("zuo_TestReflectionProjectI.TestClass"); 24             Console.WriteLine("当前类型名:{0}",Oa.Name); 25              26             MemberInfo[] minss = Oa.GetMembers(BindingFlags.Instance| 27                                                   BindingFlags.Static| 28                                                  BindingFlags.Public| 29                                                  BindingFlags.NonPublic| 30                                                  BindingFlags.DeclaredOnly);    //获取所有成员 31             Console.WriteLine(new string('-',40)); 32             Console.WriteLine("成员列表:"); 33             foreach(MemberInfo mins in minss){ 34                 Console.WriteLine("{0}",mins); 35             } 36             Console.WriteLine(new string('-',40)); 37             //实例化一个TestClass对象 38              39             //1.使用公共的构造函数实例化,带一个参数 40             //使用程序集Assembly.CreateInstance()进行实例化 41             //第一个参数:代表了要创建的类型实例的字符串名称 42             //第二个参数:说明是不是大小写无关(Ignore Case) 43             //第三个参数:在这里指定Default,意思是不使用BingdingFlags的策略(你可以把它理解成null,但是BindingFlags是值类型,所以不可能为null,必须有一个默认值,而这个Default就是它的默认值); 44             //第四个参数:是Binder,它封装了CreateInstance绑定对象(Calculator)的规则,我们几乎永远都会传递null进去,实际上使用的是预定义的DefaultBinder; 45             //第五个参数:是一个Object[]数组类型,它包含我们传递进去的参数,有参数的构造函数将会使用这些参数; 46             //第六个参数:是一个CultureInfo类型,它包含了关于语言和文化的信息(简单点理解就是什么时候ToString("c")应该显示“¥”,什么时候应该显示“$”)。 47             //第七个参数:是一个object[]数组,描述特性 48              49             //Ta就是我们实例化后的一个TestClass对象了。 50             //这是调用的公共构造函数 51             object Ta = ass.CreateInstance(Oa.FullName,true,BindingFlags.Default,null,new object[]{"麦克"},null,null); 52              53             //这是调用的private构造函数 54             //方式用的是 Activator.CreateInstance()进行实例化,返回一个ObjectHandle类的对象 55             //需要Unwrap()才能返回object对象 56             //Activator.CreateInstance()的参数说明 57             //第一个参数:当前程序集的全名称,字符串的形式 58             //第二个参数:代表了要创建的类型实例的字符串名称 59             //第三个参数:说明是不是大小写无关(Ignore Case) 60             //第四个参数:BindingFlags 61             //              Default,意思是不使用BingdingFlags的策略 62             //              NonPublic指定是非公共的类型 63             //第五个参数:是Binder,它封装了CreateInstance绑定对象(Calculator)的规则,我们几乎永远都会传递null进去,实际上使用的是预定义的DefaultBinder; 64             //第六个参数:是一个Object[]数组类型,它包含我们传递进去的参数,有参数的构造函数将会使用这些参数; 65             //其他参数:……略 66             ObjectHandle handler = Activator.CreateInstance(null, Oa.FullName,true,BindingFlags.Default| 67                                                             BindingFlags.Instance| 68                                                                BindingFlags.NonPublic,null,null,null,null,null); 69             //Tb是通过私有的构造函数创建的对象 70             object Tb = handler.Unwrap(); 71              72             //调用其方法进行做些事情 73             Console.WriteLine("调用其方法进行做些事情:"); 74              75             Oa.InvokeMember("Show",BindingFlags.InvokeMethod,null,Ta,null,null,null,null); 76             //Pshow方法是private,依然无阻力调用 77             Oa.InvokeMember("Pshow",BindingFlags.InvokeMethod|BindingFlags.Instance|BindingFlags.NonPublic,null,Ta,null,null,null,null); 78              79             Console.WriteLine(); 80             Oa.InvokeMember("Show",BindingFlags.InvokeMethod,null,Tb,null,null,null,null); 81             Oa.InvokeMember("Pshow",BindingFlags.InvokeMethod|BindingFlags.Instance|BindingFlags.NonPublic,null,Tb,null,null,null,null); 82              83             Console.WriteLine(); 84             //通过属性后台生成的方法,查阅了一次属性 85             string str = Oa.InvokeMember("get_ClassName",BindingFlags.InvokeMethod,null,Tb,null,null,null,null).ToString(); 86             Console.WriteLine("Tb属性名:{0}",str); 87              88             Console.WriteLine(); 89             //通过属性后台生成的方法,设置了一次属性,属性的set访问器是private 90             Oa.InvokeMember("set_ClassName",BindingFlags.InvokeMethod|BindingFlags.Instance|BindingFlags.NonPublic,null,Tb,new object[]{"撒旦"},null,null,null); 91              92             Console.WriteLine(); 93             //通过属性后台生成的方法,再次查阅了更改后的属性 94             str = Oa.InvokeMember("get_ClassName",BindingFlags.InvokeMethod,null,Tb,null,null,null,null).ToString(); 95             Console.WriteLine("Tb属性名:{0}",str); 96              97             Console.WriteLine(new string('-',40));             98             Console.WriteLine("属性:"); 99             str = Oa.InvokeMember("ClassName",BindingFlags.GetProperty,null,Tb,null,null,null,null).ToString();100             Console.WriteLine("{1}获取到的属性是:{0}",str,"Tb");101             102             //这个设置属性很有意思103             //整体是public,而set却是private,所以BindingFlags就需要NonPublic和Public,缺一个都不行104             Console.WriteLine();105             Oa.InvokeMember("ClassName",BindingFlags.SetProperty|BindingFlags.Instance|BindingFlags.NonPublic|BindingFlags.Public,null,Tb,new object[]{"吸血鬼"},null,null,null);106             107             Console.WriteLine();108             str = Oa.InvokeMember("ClassName",BindingFlags.GetProperty,null,Tb,null,null,null,null).ToString();109             Console.WriteLine("{1}获取到的属性是:{0}",str,"Tb");110             111             Console.WriteLine(new string('-',40));112             Console.WriteLine("直接访问或设置私有字段:");113             string field = Oa.InvokeMember("className",BindingFlags.GetField|BindingFlags.Instance|BindingFlags.NonPublic,null,Tb,null,null,null,null).ToString();114             Console.WriteLine("Tb's private string className:{0}",field);115             116             //直接设置类中的私有字段117             Oa.InvokeMember("className",BindingFlags.SetField|BindingFlags.Instance|BindingFlags.NonPublic,null,Tb,new object[]{"血之修罗"},null,null,null);118             119             Console.WriteLine("\n重新设置后:");120             field = Oa.InvokeMember("className",BindingFlags.GetField|BindingFlags.Instance|BindingFlags.NonPublic,null,Tb,null,null,null,null).ToString();121             Console.WriteLine("Tb's private string className:{0}",field);122             123             Console.WriteLine();124             Console.WriteLine("反射让你的代码无所遁形,如果你的代码没有做一些防护措施的话。");125         }126     }127     #endregion128     129     #region "待反射调用的类"130     public class TestClass131     {132         private string className;133         134         private TestClass()135         {136             this.className = "深渊恶魔";137         }138         139         public TestClass(string n){140             this.className = n;141         }142         143         public void Show(){144             Console.WriteLine("className is:{0}",this.className);145         }146         147         private void Pshow(){148             Console.WriteLine("万恶的家伙,您不应该调用此方法。它的名字是:{0}",this.className);149         }150         151         public string ClassName{152             get{ return this.className; }153             private set{ Console.WriteLine("正在调用私有的属性设置器,破坏规则的家伙真讨厌!");this.className=value; }154         }        155         156     }157     #endregion158 }

代码运行结果:

D:\A>OtherRoad
先载入一个程序集,然后分析它的结构:
程序集的名称是:OtherRoad, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null
----------------------------------------
类型名:Program
类型名:TestClass
----------------------------------------
当前类型名:TestClass
----------------------------------------
成员列表:
Void Show()
Void Pshow()
System.String get_ClassName()
Void set_ClassName(System.String)
Void .ctor()
Void .ctor(System.String)
System.String ClassName
System.String className
----------------------------------------
调用其方法进行做些事情:
className is:麦克
万恶的家伙,您不应该调用此方法。它的名字是:麦克

className is:深渊恶魔
万恶的家伙,您不应该调用此方法。它的名字是:深渊恶魔

Tb属性名:深渊恶魔

正在调用私有的属性设置器,破坏规则的家伙真讨厌!

Tb属性名:撒旦
----------------------------------------
属性:
Tb获取到的属性是:撒旦

正在调用私有的属性设置器,破坏规则的家伙真讨厌!

Tb获取到的属性是:吸血鬼
----------------------------------------
直接访问或设置私有字段:
Tb's private string className:吸血鬼

重新设置后:
Tb's private string className:血之修罗

.NET 反射的更多相关文章

  1. 隐私泄露杀手锏 —— Flash 权限反射

    [简版:http://weibo.com/p/1001603881940380956046] 前言 一直以为该风险早已被重视,但最近无意中发现,仍有不少网站存在该缺陷,其中不乏一些常用的邮箱.社交网站 ...

  2. Java学习之反射机制及应用场景

    前言: 最近公司正在进行业务组件化进程,其中的路由实现用到了Java的反射机制,既然用到了就想着好好学习总结一下,其实无论是之前的EventBus 2.x版本还是Retrofit.早期的View注解框 ...

  3. 关于 CSS 反射倒影的研究思考

    原文地址:https://css-tricks.com/state-css-reflections 译者:nzbin 友情提示:由于演示 demo 的兼容性,推荐火狐浏览.该文章篇幅较长,内容庞杂,有 ...

  4. 编写高质量代码:改善Java程序的151个建议(第7章:泛型和反射___建议106~109)

    建议106:动态代理可以使代理模式更加灵活 Java的反射框架提供了动态代理(Dynamic Proxy)机制,允许在运行期对目标类生成代理,避免重复开发.我们知道一个静态代理是通过主题角色(Prox ...

  5. 运用Mono.Cecil 反射读取.NET程序集元数据

    CLR自带的反射机智和API可以很轻松的读取.NET程序集信息,但是不能对程序集进行修改.CLR提供的是只读的API,但是开源项目Mono.Cecil不仅仅可以读取.NET程序集的元数据,还可以进行修 ...

  6. .NET面试题系列[6] - 反射

    反射 - 定义,实例与优化 在面试中,通常会考察反射的定义(操作元数据),可以用反射做什么(获得程序集及其各个部件),反射有什么使用场景(ORM,序列化,反序列化,值类型比较等).如果答得好,还可能会 ...

  7. .NET基础拾遗(4)委托、事件、反射与特性

    Index : (1)类型语法.内存管理和垃圾回收基础 (2)面向对象的实现和异常的处理基础 (3)字符串.集合与流 (4)委托.事件.反射与特性 (5)多线程开发基础 (6)ADO.NET与数据库开 ...

  8. C++的性能C#的产能?! - .Net Native 系列五:.Net Native与反射

    此系列系小九的学堂原创翻译,翻译自微软官方开发向导,一共分为六个主题.本文是第五个主题:.Net Native与反射. 向导文链接:<C++的性能C#的产能?! - .Net Native 系列 ...

  9. [源码]Literacy 快速反射读写对象属性,字段

    Literacy 说明 Literacy使用IL指令生成方法委托,性能方面,在调用次数达到一定量的时候比反射高很多 当然,用IL指令生成一个方法也是有时间消耗的,所以在只使用一次或少数几次的情况,不但 ...

  10. SI与EMI(一) - 反射是怎样影响EMI

    Mark为期两天的EMC培训中大概分成四个时间差不多的部分,简单来说分别是SI.PI.回流.屏蔽.而在信号完整性的书籍中,也会把信号完整性分为:1.信号自身传输的问题(反射,损耗):2.信号与信号之间 ...

随机推荐

  1. install命令和cp命令的区别

    基本上,在Makefile里会用到install,其他地方会用cp命令. 它们完成同样的任务——拷贝文件,它们之间的区别主要如下: 1.最重要的一点,如果目标文件存在,cp会先清空文件后往里写入新文件 ...

  2. 语法错误: 标识符“acosf”

    1>e:\vs2010\vc\include\cmath(19): error C2061: 语法错误: 标识符“acosf” 1>e:\vs2010\vc\include\cmath(1 ...

  3. java 多线程系列基础篇(十)之线程优先级和守护线程

    1. 线程优先级的介绍 java 中的线程优先级的范围是1-10,默认的优先级是5.“高优先级线程”会优先于“低优先级线程”执行. java 中有两种线程:用户线程和守护线程.可以通过isDaemon ...

  4. Mybatis 内置 Java 类型别名与 typeHandlers

    aliases There are many built-in type aliases for common Java types. They are all case insensitive, n ...

  5. 数据存储的两种方式:Cookie 和Web Storage(转)

    数据存储的两种方式:Cookie 和Web Storage   数据存储的两种方式:Cookie 和Web Storage 1.Cookie Cookie的作用就像你去超市购物时,第一次给你办张购物卡 ...

  6. cocos2dx中替代goto的用法:do{}while(0)和CC_BREAK_IF

    我们时常会调用某个函数来创建一个对象,但由于内存不足或其他异常情况发生时对象可能会创建失败,创建失败我们就要结束当前程序转到错误处理地方去处理错误或释放已生成的对象. int* p1 = new in ...

  7. Blocks UVA - 10559

    传送门 题目大意 有n个带有颜色的方块,没消除一段长度为x的连续的相同颜色的方块可以得到x^2的分数,让你用一种最优的顺序消除所有方块使得得分最多. 分析 首先不难看出这是一个区间dp,于是我们考虑如 ...

  8. vue 之 计算属性和侦听器

    计算属性 模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的.在模板中放入太多的逻辑会让模板过重且难以维护.例如: <div> {{ message.split('').rever ...

  9. Mat 与 IplImage 和 CvMat 的转换

    在 OpenCV 2 中虽然引入了方便的 Mat 类,出于兼容性的考虑,OpenCV 依然是支持 C 语言接口的 IplImage 和 CvMat 结构.如果你要与以前的代码兼容,将会涉及 Mat 与 ...

  10. dreamweaver cs5 快捷键

    撤销上一步:ctrl + Z: 回复上一步:ctrl + Y: 代码缩进:左下角(应用原格式)