Jester Zhu

从学习中得到乐趣,从乐趣中得到灵感,从灵感中创造真知。Think well,just do it.
  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

.net基础学习(三):什么是反射?

Posted on 2009-09-07 16:03  Jester Zhu  阅读(1383)  评论(0编辑  收藏  举报

一、反射的概念 :

反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力。这一概念的提出很快引发了计算机科学领域关于应用反射性的研究。它首先被程序语言的设计领域所采用,并在Lisp和面向对象方面取得了成绩。其中LEAD/LEAD++ 、OpenC++ 、MetaXa和OpenJava等就是基于反射机制的语言。最近,反射机制也被应用到了视窗系统、操作系统和文件系统中。
反射本身并不是一个新概念,它可能会使我们联想到光学中的反射概念,尽管计算机科学赋予了反射概念新的含义,但是,从现象上来说,它们确实有某些相通之处,这些有助于我们的理解。在计算机科学领域,反射是指一类应用,它们能够自描述和自控制。也就是说,这类应用通过采用某种机制来实现对自己行为的描述(self-representation)和监测(examination),并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。可以看出,同一般的反射概念相比,计算机科学领域的反射不单单指反射本身,还包括对反射结果所采取的措施。所有采用反射机制的系统(即反射系统)都希望使系统的实现更开放。可以说,实现了反射机制的系统都具有开放性,但具有开放性的系统并不一定采用了反射机制,开放性是反射系统的必要条件。一般来说,反射系统除了满足开放性条件外还必须满足原因连接(Causally-connected)。所谓原因连接是指对反射系统自描述的改变能够立即反映到系统底层的实际状态和行为上的情况,反之亦然。开放性和原因连接是反射系统的两大基本要素。
 

反射,Reflection,通过它我们可以在运行时获得各种信息,如程序集、模块、类型、字段、属性、方法和事件

通过对类型动态实例化后,还可以对其执行操作

一般用于插件式框架程序和设计模式的实现,当然反射是一种手段可以充分发挥其能量来完成你想做的任何事情(前面好象见过一位高人用反射调用一个官方类库中未说明的函数。。。)

示例:

using System;
using System.Collections.Generic;
using System.Text;

namespace Example25Lib
{
       public class Class1
       {
           private string name;
           private int age;

           //如果显式的声明了无参数构造函数,客户端只需要用程序集的CreateInstance即可实例化该类
           //在此特意不实现,以便在客户调用端体现构造函数的反射实现
           //public Class1()
           //{
           //}
           public Class1(string Name, int Age)
           {
               name = Name;
               age = Age;
           }
           public void ChangeName(string NewName)
           {
               name = NewName;
           }
           public void ChangeAge(int NewAge)
           {
               age = NewAge;
           }
           public override string ToString()
           {
               return string.Format("Name: {0}, Age: {1}", name, age);
           }
       }
}
反射实例化对象并调用其方法,属性和事件的反射调用略去

using System;
using System.Collections.Generic;
using System.Text;

//注意添加该反射的命名空间
using System.Reflection;

namespace Example25
{
       class Program
       {
           static void Main(string[] args)
           {
               //加载程序集
               Assembly tmpAss = Assembly.LoadFile(AppDomain.CurrentDomain.BaseDirectory + "Example25Lib.dll");

               //遍历程序集内所有的类型,并实例化
               Type[] tmpTypes = tmpAss.GetTypes();
               foreach (Type tmpType in tmpTypes)

     {
                   //获取第一个类型的构造函数信息
                   ConstructorInfo[] tmpConsInfos = tmpType.GetConstructors();
                   foreach (ConstructorInfo tmpConsInfo in tmpConsInfos)
                   {
                       //为构造函数生成调用的参数集合
                       ParameterInfo[] tmpParamInfos = tmpConsInfo.GetParameters();
                       object[] tmpParams = new object[tmpParamInfos.Length];
                       for (int i = 0; i < tmpParamInfos.Length; i++)
                       {
                           tmpParams[i] = tmpAss.CreateInstance(tmpParamInfos[i].ParameterType.FullName);
                           if (tmpParamInfos[i].ParameterType.FullName == "System.String")
                           {
                               tmpParams[i] = "Clark";
                           }
                       }

                       //实例化对象
                       object tmpObj = tmpConsInfo.Invoke(tmpParams);
                       Console.WriteLine(tmpObj);

                       //获取所有方法并执行
                       foreach (MethodInfo tmpMethod in tmpType.GetMethods())


                       {
                           //为方法的调用创建参数集合
                           tmpParamInfos = tmpMethod.GetParameters();
                           tmpParams = new object[tmpParamInfos.Length];
                           for (int i = 0; i < tmpParamInfos.Length; i++)
                           {
                               tmpParams[i] = tmpAss.CreateInstance(tmpParamInfos[i].ParameterType.FullName);
                               if (tmpParamInfos[i].ParameterType.FullName == "System.String")
                               {
                                   tmpParams[i] = "Clark Zheng";
                               }
                               if (tmpParamInfos[i].ParameterType.FullName == "System.Int32")
                               {
                                   tmpParams[i] = 27;
                               }
                           }
                           tmpMethod.Invoke(tmpObj, tmpParams);

             }

                       //调用完方法后再次打印对象,比较结果
                       Console.WriteLine(tmpObj);
                   }
               }

               Console.ReadLine();
           }
       }
}
结果:
Name: Clark, Age: 0
Name: Clark Zheng, Age: 27