反射是一种晚绑定,它可以被开发者用来设计出更具灵活性的代码,而代价则是花费更多的系统资源开销使得应用程序可以在运行时获取一些未知信息。
说白了,在编写代码的时候,开发者可能还不知道或不能确定一些对象的信息,于是把决定权交给代码本身,将来在需要的时候由代码自己去获取和判断这些信息并作出相应的反应。这样的方式固然可以使代码更加灵活,但在想要使用反射的时候,必须先考虑好在性能与灵活之间的一个权衡,不能盲目地因为C#提供了反射机制就一个劲儿地用,我们应该发扬John Carmack“榨干PC机3D图像性能”的精神,而不要因为硬件越来越便宜就让机器背负沉重的包袱去做一些意义不大的事情。
我们还是继续本系列一贯的风格,用生活中的例子来讲解什么是反射以及反射的基本使用方式。
我先问大家一个问题:“《集结号》中谷子地穿的南朝鲜军服从哪儿来?”显然不是临时找裁缝赶制的,肯定是从俘虏身上扒下来的。好,那我们就来补充一点儿《集结号》的镜头看看什么是反射。
一天傍晚,一名又饥又渴的南朝鲜掉队士兵在乡间小路上蹒跚地走着,他甚至都快扛不动身上的枪了,也不知道队伍在哪里,只好认准一个方向努力寻找。“不许动!”从路边草丛中跳出的两名埋伏着的中国士兵喝道,这名南朝鲜士兵本来就精疲力尽,哪受得了这番惊吓,当时就一屁股坐在地上了。中国士兵把他押回营部并向上级报告,他们在附近巡逻的时候意外俘获一名南朝鲜掉队士兵,听候处置。
这件事情被上报到王团长那里,王团长正拿着放大镜跟几名军官在地图前研究作战方案,听到这件事,不紧不慢地跟他身边的高连长说道:“高诚啊,你去看看,没啥大问题就送到后方战俘营去。”“是!”高连长接到命令便转身出去了。
高连长来到一件狭小的房间,那名南朝鲜士兵正半躺在地上,手里拿着中国士兵给他的馒头和水,一口一口地嚼着,看到高连长进来了,眼睛里流露出紧张恐惧的神情。高连长简单介绍了中国方面善待俘虏的政策,便开始审问他——注意,反射开始了!
“哪儿的?”“联合国军李承晚系部队3团2排1班。”
“叫什么名字?”“思密达。”
“这次执行什么任务?”“潜入贵军阵地并指引炮兵进行射击。”
……
一番软硬兼施之后,高连长掌握了这个小兵所有的信息,整理好材料去跟王团长汇报去了,而这个小兵被带到一个小屋子里继续啃馒头去了,身上的行头也被换了下来,换上了专门为战俘准备的棉衣。
高连长一手拿着笔录材料,一手拎着南朝鲜士兵的全套行头去见了王团长,汇报之后王团长会心地笑了:“小高啊,不错,你父亲当初把你交给我,我就叫他放心,你是块好料子,怎么样,我没说错吧,呵呵,不过你也莫要骄傲,我们不能轻敌,这样,你带着这身行头去找谷子地,叫他换上之后连夜潜入敌方阵地,敌人不是想拿炮弹砸我们吗,那我们就用他们的方式,先发制人,让老谷指引我军炮火明日凌晨发起总攻。”
谷子地换上了这身洋行头,揣了半条烟就出发了,他潜入敌方阵地后冷静地掏出敌人的望远镜进行了周密地观察,并通过无线电给后方友军传递射击参数,成功地引导我军取得了这次重大胜利。
上面的故事情节描述得有点儿多了,赶紧来看代码吧,首先,我们得有个倒霉的南朝鲜士兵做引子:
 1: namespace UN
2: {
3:     internal class SouthKoreaArmy
4:     {
5:         public string Name { get; set; }
6: 
7:         public string Search(string destination)
8:         {
 9:             return "OK, fire!";
10:         }
11:     }
12: }
然后,我们用代码来实现谷子地所完成的行动:
1: internal class ChineseArmy
2: {
3:     public static void Action(object soldier)
4:     {
5:         Type type = soldier.GetType();
 6:         object ziDiGu = Activator.CreateInstance(type);
7:         MethodInfo methodInfo = type.GetMethod("Search");
8:
9:         Console.WriteLine(methodInfo.Invoke(ziDiGu, new object[] { "Position" }));
10:     }
11: }
这段代码很好理解,第5行我们审问了被俘的南朝鲜士兵并得到了他提供的信息和服装,然后让老谷用他的服装化装成南朝鲜士兵。第7行,小样,你不是想侦查我吗,好,咱就来个以牙还牙,用你的方式来搞定你!接下来,谷子地潜入地方阵地并引导了我军发起总攻。
这里展示了反射的一些基本应用,例如获取类型信息、利用获取的类型动态生成对象,并动态调用其方法。当然,反射机制能做的事情不仅仅是这几样,不过目标都一样,由程序自己去获取信息、做出反应。在不用担心性能开销的情况下,反射可以使你的程序更加灵活强大!
白话C#反射_C#