反射原理及简介

一.什么是反射

 Reflection,中文翻译为反射。这是.Net中获取运行时类型信息的方式,

.Net的应用程序由几个部分:‘程序集(Assembly)’、‘模块(Module)’、‘类型(class)’组成,而反射提供一种编程的方式,让程序员可以在程序运行期获得这几个组成部分的相关信息,

例如:Assembly类可以获得正在运行的程序集信息,也可以动态的加载程序集,以及在程序集中查找类型信息,并创建该类型的实例。

Type类可以获得对象的类型信息,此信息包含对象的所有要素:方法、构造器、属性等等,通过Type类可以得到这些要素的信息,并且调用之。

MethodInfo包含方法的信息,通过这个类可以得到方法的名称、参数、返回值等,并且可以调用之。诸如此类,

还有FieldInfo、EventInfo等等,这些类都包含在System.Reflection命名空间下。

 二. 关于程序集和命名空间的关系

很多人对这个概念可能还是很不清晰,对于合格的.Net程序员,有必要对这点进行澄清。

程序集是.NET应用程序执行的最小单元,编译出来的.dll和.exe都是程序集。 程序集和命名空间的关系不是一一对应,也不互相包含,一个程序集里面可以有多个命名空间,一个命名空间也可以在多个程序中存在,这样说可能有点模糊,举个例子:

程序集A包含两个命名空间:

namespace  N1
{
      public  class  AC1  {…}
      public  class  AC2  {…}
}
namespace  N2
{
      public  class  AC3  {…}
      public  class  AC4{…}
}

程序集B包含两个命名空间:

namespace  N1
{
      public  class  BC1  {…}
      public  class  BC2  {…}
}
namespace  N2
{
      public  class  BC3  {…}
      public  class  BC4{…}
}

这两个程序集中都有N1和N2两个命名空间,而且各声明了两个类,这样是完全可以的,然后我们在一个应用程序中引用程序集A,那么在这个应用程序中,我们能看到N1下面的类为AC1和AC2,N2下面的类为AC3和AC4。

接着我们去掉对A的引用,加上对B的引用,那么我们在这个应用程序下能看到的N1下面的类变成了BC1和BC2,N2下面也一样。 如果我们同时引用这两个程序集,那么N1下面我们就能看到四个类:AC1、AC2、BC1和BC2。

到这里,我们可以清楚一个概念了,命名空间只是说明一个类型是那个族的,比如有人是汉族、有人是回族;而程序集表明一个类型住在哪里,比如有人住在北京、有人住在上海;那么北京有汉族人,也有回族人,上海有汉族人,也有回族人,这是不矛盾的。

上面我们说了,程序集是一个类型居住的地方,那么在一个程序中要使用一个类,就必须告诉编译器这个类住在哪儿,编译器才能找到它,也就是说必须引用该程序集。

 那么如果在编写程序的时候,也许不确定这个类在哪里,仅仅只是知道它的名称,就不能使用了吗?答案是可以,这就是反射了,就是在程序运行的时候提供该类型的地址,而去找到它。

 

 三.为什么使用反射

 有人会有疑问,程序所用的类既然可以事先写好,那么为什么还要在程序运行的时候去生成,这样岂不是浪费系统资源。存在就是合理的,既然微软给我们开发这项技术,肯定是这个这个东西有需求,举个简单的例子:

现在要开发一个报表打印模块,有的企业要求数据以Excel报表,有的企业要求打印水晶报表等等,这个时候我们就可以先定义一个接口,任何报表打印方法都必须实现这个接口:

 

   public interface  IReport
    {
        void StartPrint();
    }

我们通过配置文件可以加载对应的读取报表的类型,因为报表打印类都实现了IReport的接口,所以都可以通过反射的方式强转为该接口类型,这样就可以实现无需修改底层代码就可以实现打印不同的数据报表,符合我i们程序设计的开闭原则,这个就是反射最经典的应用。

public static class Factory
{
//【1】读取配置文件
static string reportType = ConfigurationManager.AppSettings["ReportType"].ToString();
//【2】使用反射创建实现接口类的对象并以接口类型返回
public static IReport ChooseReportType()
{
return (IReport)Assembly.Load("UseFactory").CreateInstance("UseFactory." + reportType);
}
}

四.如何使用反射获取类型

获取类信息有两种方式:

第一种方式,得到实例化对象。 这个时侯我仅仅是得到这个实例对象,得到的方式也许是一个object的引用,也许是一个接口的引用,但是我并不知道它的确切类型,我需要了解,那么就可以通过调用System.Object上声明的方法GetType来获取实例对象的类型对象,比如在某个方法内,我需要判断传递进来的参数是否实现了某个接口,如果实现了,则调用该接口的一个方法。

   public void Progress(object o)
        {
            Type objType = o.GetType();

            if (objType.GetInterface("ITest") !=null)
            {
                //调用该接口的方法
            }
        }

第二种获取类型的方法是通过Type.GetType以及Assembly.GetType方法,但是在使用该方法时我们需要注意一些问题点。在程序集A.dll中需要反射程序集B.dll中的类型。如果使用稍有不慎,就会产生运行时错误。例如使用Type.GetType("BNameSpace.ClassName")在程序集A.dll获取程序集B.dll中的类型,就会返回Null。

关于跨程序集的反射,有两点需要注意:

1、如果使用typeof,编译能通过,则跨程序集的反射一定可以正常运行。可以说,typeof是支持强类型的。比如

 Type supType = typeof(BNameSpace.SubSpace.Class);

如果当前程序集没有添加对EnterpriseServerBase.dll的引用,则编译会报错。

2、如果使用Type.GetType来进行反射的话,情况就复杂些。这是因为Type.GetType是非强类型的。Type.GetType的参数是一个string为类型的完全限定名,如果当string表示的目标类型不在当前程序集中,则运行时Type.GetType会返回null。解决的办法是:首先加载目标程序集,然后再使用Assembly.GetType方法来获取类型。如:

Assembly asmb = Assembly.LoadFrom("EnterpriseServerBase.dll") ;
Type supType = asmb.GetType("EnterpriseServerBase.DataAccess.IDBAccesser") ;

注意:当使用Type.GetType的时候,即使你添加了对EnterpriseServerBase.dll的引用,Type.GetType("EnterpriseServerBase.DataAccess.IDBAccesser")也会返回null,这是因为Type.GetType只会在当前程序集中进行类型搜索。

5.如何根据类型动态创建对象

第一种方式

public class Example
{
    static void Main()
    {
        // Create an instance of the StringBuilder type using 
        // Activator.CreateInstance.
        Object o = Activator.CreateInstance(typeof(StringBuilder));

        // Append a string into the StringBuilder object and display the 
        // StringBuilder.
        StringBuilder sb = (StringBuilder) o;
        sb.Append("Hello, there.");
        Console.WriteLine(sb);

        // Create an instance of the SomeType class that is defined in this 
        // assembly.
        System.Runtime.Remoting.ObjectHandle oh = 
            Activator.CreateInstanceFrom(Assembly.GetEntryAssembly().CodeBase, 
                                         typeof(SomeType).FullName);

        // Call an instance method defined by the SomeType type using this object.
        SomeType st = (SomeType) oh.Unwrap();

        st.DoSomething(5);
    }
}

 

例二:根据有参数的构造器创建对象

namespace  TestSpace  
{
  public  class  TestClass
      {
      private  string  _value;
      public  TestClass(string  value)  
    {
      _value=value;
      }
  }
}

Type  t  =  Type.GetType(“TestSpace.TestClass”);
Object[]  constructParms  =  new  object[]  {“hello”};  //构造器参数
TestClass  obj  =  (TestClass)Activator.CreateInstance(t,constructParms);

把参数按照顺序放入一个Object数组中即可

第三种方式:

假如我们并没有加载对应.cs文件的程序集,那么我们该如何简洁的创建对象类型呢:

    object objCal =Assembly.LoadFrom("CalDLL.dll").CreateInstance("CalDLL.Calculator");

是不是很简洁,当然我们也可以先获取对象的类型,再用Activator.CreateInstance 来创建对象,不过稍显麻烦:

 Assembly objAssembly = Assembly.LoadFrom("CalDLL.dll");


 Type objType = objAssembly.GetType("CalDLL.Calculator");


 object objCal = Activator.CreateInstance(objType);

这样也能达到相同的效果。

 

6.如何获取方法以及动态调用方法

应用反射动态调用方法的示例如下:

using System;
using System.Reflection;

class Program
{
    // Methods to get:

    public void MethodA(int i, int j) { }

    public void MethodA(int[] i) { }

    public unsafe void MethodA(int* i) { }

    public void MethodA(ref int r) {}

    // Method that takes an out parameter:
    public void MethodA(int i, out int o) { o = 100;}


  static void Main(string[] args)
  {
    MethodInfo mInfo;

    // Get MethodA(int i, int j)
    mInfo = typeof(Program).GetMethod("MethodA",
        BindingFlags.Public | BindingFlags.Instance,
        null,
        CallingConventions.Any,
        new Type[] { typeof(int), typeof(int) },
        null);
    Console.WriteLine("Found method: {0}", mInfo);

    // Get MethodA(int[] i)
    mInfo = typeof(Program).GetMethod("MethodA",
        BindingFlags.Public | BindingFlags.Instance,
        null,
        CallingConventions.Any,
        new Type[] { typeof(int[]) },
        null);
    Console.WriteLine("Found method: {0}", mInfo);

    // Get MethodA(int* i)
    mInfo = typeof(Program).GetMethod("MethodA",
        BindingFlags.Public | BindingFlags.Instance,
        null,
        CallingConventions.Any,
        new Type[] { typeof(int).MakePointerType() },
        null);
    Console.WriteLine("Found method: {0}", mInfo);

    // Get MethodA(ref int r)
    mInfo = typeof(Program).GetMethod("MethodA",
        BindingFlags.Public | BindingFlags.Instance,
        null,
        CallingConventions.Any,
        new Type[] { typeof(int).MakeByRefType() },
        null);
    Console.WriteLine("Found method: {0}", mInfo);

    // Get MethodA(int i, out int o)
    mInfo = typeof(Program).GetMethod("MethodA",
        BindingFlags.Public | BindingFlags.Instance,
        null,
        CallingConventions.Any,
        new Type[] { typeof(int), typeof(int).MakeByRefType() },
        null);
    Console.WriteLine("Found method: {0}", mInfo);

  }
}
关于GetMethod()的更多信息可以通过如下:

https://docs.microsoft.com/en-us/dotnet/api/system.type.getmethod?view=netframework-4.8

7.如何动态的获取属性信息:

下面的例子简单的说明了如何获取属性信息

 void GetProperty()
        {
            Type objType = typeof(string);
           PropertyInfo[]  obj =  objType.GetProperties();
           if (obj !=null)
           {
               foreach (var item in obj)
               {
                   this.textBox1.Text = this.textBox1.Text + item.Name.ToString() + "\r\n";
               }
           }
        }

 

posted @ 2019-06-17 17:11  大艺术家007  阅读(1130)  评论(0编辑  收藏  举报