C#反射之Assembly.Load,Assembly.LoadFile 与 Assembly.LoadFrom方法介绍

1、C#反射是什么?

Reflection,中文翻译为反射。

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

Assembly类可以获得正在运行的装配件信息,也可以动态的加载装配件,以及在装配件中查找类型信息,并创建该类型的实例。

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

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

 

C#反射命名空间详细介绍:

1.System.Reflection命名空间内的各类型

(1) Assembly通过它可以加载、了解和操纵一个程序集

(2) AssemblyName 通过它可以找到大量隐藏在程序集的身份中的信息,如版本信息、区域信息等

(3) EventInfo  事件的信息

(4) FieldInfo  字段的信息

(5) MethodInfo  方法的信息

(6) ParameterInfo  参数的信息

(7) PropertyInfo  属性的信息

(8) MemberInfo  是抽象基类,为  EventInfo、FieldInfo 、MethodInfo、PropertyInfo等类型定义了公共的行为。

(9) Module 用来访问带有多文件程序集的给定模块 

2.System.Type类

System.Type支持的成员可以分为这样几类

(1) Is***   用来检查一个类型的元数据,如IsAbstract、IsClass、IsValueType等等

(2) Get*** 用来从类型得到指定项目,如GetEvent()得到类型的一个指定的事件(EventInfo)。
另外,这些方法都有一个单数版本和一个复数版本。如GetEvent()对应有一个复数版本GetEvents(),
该方法返回一个相关的EventInfo数组。

(3) FindMembers()   根据查询条件返回一个MemberInfo类型的数组

(4)GetType()  该静态方法根据一个字符串名称返回一个Type实例

(5)InvokeMember()  对给定项目进行晚期绑定

3.得到一个Type类型实例的三种方法

因为Type是一个抽象类,所以不能直接使用new关键字创建一个Type对象

(1)使用System.Object.GetType()

Person pe=new Person();    //---------定义pe为person类的一个对象  
Type t=pe.GetType(); 

(2)使用System.Type.GetType()静态方法,参数为类型的完全限定名

Type t=Type.GetType("Entity.Person"); 该方法被重载,允许指定两个布尔类型的参数,一个用来控制当前类型不能找到时是否抛出异常,

另一个用来指示是否区分字符串大小写

Type t=Type.GetType("Entity.Person",false,true); 注意到传入的字符串并没有包含类型所在的程序集信息,此时该类型便被认为是定义在当前执行的程序集中的。

要得到一个外部私有程序集的类型元数据时,字符串参数必须使用类型完全限定名加上类型所在程序集的友好名字

Type t=Type.GetType("Entity.Person","Entity");  //------"Entity"即为类型所在程序集的友好名字  嵌套类型:传入的字符串可以指定一个+标记来表示一个嵌套类型,

如希望得到一个嵌套在person类中的枚举类型City的类型信息,则可以这样

Type t=Type.GetType("Entity.person+City");

(3)使用typeof运算符

Type  t=typeof(person); 三种方法的比较:

使用第一种方法必须先建立一个实例,而后两种方法不必先建立实例。但使用typeof运算符仍然需要知道类型的编译时信息,

而使用System.Type.GetType()静态方法不需要知道类型的编译时信息,所以是首选方法。

一个最简单的C#反射实例,首先编写类库如下:
namespace ReflectionTest

   public class WriteTest 
   { 
       //带参数的公共方法
    public void WriteString(string s, int i) 
    { 
       Console.WriteLine("WriteString:" + s + i.ToString()); 
    }  
    //带一个参数的静态方法
    public static void StaticWriteString(string s)
    {
          Console.WriteLine("StaticWriteString:" + s);
    } 
    //不带参数的静态方法
    public static void NoneParaWriteString()
    {
         Console.WriteLine("NoParaWriteString");
    } 
   }
}

class TestApp

   public static void Main() 
   { 
    Assembly ass; 
    Type type; 
    Object obj;  
    //用来测试静态方法
    Object any = new Object();  
    //指定类库文件必须使用绝对路径,不能使用相对路径
    ass = Assembly.LoadFile(@"D:/Source Code/00.C#Sudy/01.Reflection/01/ReflectTest.dll"); 
    //命名空间和类的名字必须一起指定 
    type = ass.GetType("ReflectionTest.WriteTest"); 
   
    /**//*example1---------*/ 
    MethodInfo method = type.GetMethod("WriteString");  
    string test = "test"; 
    int i = 1;  
    Object[] parametors = new Object[]{test,i};  
    //在例子1种必须实例化反射要反射的类,因为要使用的方法并不是静态方法。
    //创建对象实例
    obj = ass.CreateInstance("ReflectionTest.WriteTest");     
    //执行带参数的公共方法
    method.Invoke(obj, parametors); 
    //method.Invoke(any, parametors);//异常:必须实例化反射要反射的类,因为要使用的方法并不是静态方法。
   
    /**//*example2----------*/  
    method = type.GetMethod("StaticWriteString"); 
    method.Invoke(null, new string[] { "test"});  //第一个参数忽略
    //对于第一个参数是无视的,也就是我们写什么都不会被调用,
    //即使我们随便new了一个any这样的Object,当然这种写法是不推荐的。
    //但是对应在例子1种我们如果Invoke的时候用了类型不一致的实例来做为参数的话,将会导致一个运行时的错误。
    method.Invoke(obj, new string[] { "test"});
    method.Invoke(any, new string[] { "test"});
   
    /**//*example3-----------*/ 
    method = type.GetMethod("NoneParaWriteString"); //调用无参数静态方法的例子,这时候两个参数我们都不需要指定,用null就可以了。s 
    method.Invoke(null, null);
   }

 

先来讲解Assembly.Load方法,该方法会有多个重载版本,其中一个就是提供程序集的详细信息,即程序集的标识,包括程序集的名称,版本,区域信息,公有密钥标记,全部都是以一个字符串的形式提供,例如:"MyAssembly,Version=1.0.0.0,culture=zh-CN,PublicKeyToken=47887f89771bc57f”.

那么,使用Assembly.Load加载程序集的顺序是怎样的呢?首先它会去全局程序集缓存查找,然后到应用程序的根目录查找,最后会到应用程序的私有路径查找。

当然,如果你使用的是弱命名程序集,也即只给出程序集的名称,那么这个时候,CLR将不会在程序集上应用任何安全或者部署策略,而且Load也不会到全局缓存程序集中查找程序集。

 

上应用安全和部署策略,推荐使用该方法动态加载程序集,至于LoadFrom和LoadWithPartialName。

首先我们还是来看看LoadFrom方法,这个方法的原理是这样的:我们如果要使用它来动态加载程序集,必须告诉它程序集的路径,也即在哪个目录下面,CLR会去加载与你指定的路径完全匹配的程序集。记住,当我们指定程序集路径时,不能包括任何关于程序集强命名的信息,所以,CLR不会在我们指定的程序集文件上应用任何策略,而且也不会去任何其他的地方搜索程序集,简言之,它就是指哪打哪,呵呵。

例如:你有个程序集在D:/Test/MyAssembly.dll,你要用Assembly.LoadFrom加载该程序集,代码就如下:

Assembly assembly = Assembly.LoadFrom(@”D:/Test/MyAssembly.dll”);

对于,LoadWithParitalName方法,推荐大家最好不要使用它,因为程序无法确定最终要去加载哪个程序集的版本,所以我们这里只是简单的介绍一下它的工作原理:你可以传递一个程序集标识给它,包括程序集名称,至于其他信息是可选的(区域信息,公有密钥等),该方法执行时,会首先检查应用程序中配置文件的qualifyAssembly节点,如果存在,则把该部分名称的程序集替换成完全的程序集标识,如果不存在,则使用程序集名称先到应用程序根目录下查找,然后是私有目录,没有找到的话,就到程序集全局缓存中查找。简单过程如下:

       应用程序根目录 -> 应用程序私有目录 -> 程序集全局缓存.

Assembly.Load()方法,Assembly.LoadFrom()方法,Assembly.LoadFile()方法的区别!

1,Assembly.Load()

这个方法通过程序集的长名称(包括程序集名,版本信息,语言文化,公钥标记)来加载程序集的,会加载此程序集引用的其他程序集,一般情况下都应该优先使用 这个方法,他的执行效率比LoadFrom要高很多,而且不会造成重复加载的问题(原因在第2点上说明)

使用这个方法的时候, CLR会应用一定的策略来查找程序集,实际上CLR按如下的顺序来定位程序集:

⑴如果程序集有强名称,在首先在全局程序集缓(GAC)中查找程序集。         

⑵如果程序集的强名称没有正确指定或GAC中找不到,那么通过配置文件中的<codebase>元素指定的URL来查找

⑶如果没有指定强名称或是在GAC中找不到,CLR会探测特定的文件夹:

假设你的应用程序目录是C:/AppDir,<probing>元素中的privatePath指定了一个路径Path1,你要定位的程序集是AssemblyName.dll则CLR将按照如下顺序定位程序集

          C:/AppDir/AssemblyName.dll

          C:/AppDir/AssemblyName/AssemblyName.dll

          C:/AppDir/Path1/AssemblyName.dll

          C:/AppDir/Path1/AssemblyName/AssemblyName.dll

如果以上方法不能找到程序集,会发生编译错误,如果是动态加载程序集,会在运行时抛出异常!

2,Assembly.LoadFrom()

这个方法从指定的路径来加载程序集,实际上这个方法被调用的时候,CLR会打开这个文件,获取其中的程序集版本,语言文化,公钥标记等信息,把他们传递给 Load方法,接着,Load方法采用上面的策略来查找程序集。如果找到了程序集,会和LoadFrom方法中指定的路径做比较,如果路径相同,该程序集 会被认为是应用程序的一部分,如果路径不同或Load方法没有找到程序集,那该程序集只是被作为一个“数据文件”来加载,不会被认为是应用程序的一部分。 这就是在第1点中提到的Load方法比LoadFrom方法的执行效率高的原因。另外,由于可能把程序集作为“数据文件”来加载,所以使用 LoadFrom从不同路径加载相同程序集的时候会导致重复加载。当然这个方法会加载此程序集引用的其他程序集。

3,Assembly.LoadFile()

这个方法是从指定的文件来加载程序集,和上面方法的不同之处是这个方法不会加载此程序集引用的其他程序集!

结论:一般大家应该优先选择Load方法来加载程序集,如果遇到需要使用LoadFrom方法的时候,最好改变设计而用Load方法来代替!

另:Assembly.LoadFile 与 Assembly.LoadFrom的区别

1、Assembly.LoadFile只载入相应的dll文件,比如Assembly.LoadFile("abc.dll"),则载入abc.dll,假如abc.dll中引用了def.dll的话,def.dll并不会被载入。

Assembly.LoadFrom则不一样,它会载入dll文件及其引用的其他dll,比如上面的例子,def.dll也会被载入。

2、用Assembly.LoadFrom载入一个Assembly时,会先检查前面是否已经载入过相同名字的Assembly,比如abc.dll有两个版本(版本1在目录1下,版本2放在目录2下),程序一开始时载入了版本1,当使用Assembly.LoadFrom("2//abc.dll")载入版本2时,不能载入,而是返回版本1。Assembly.LoadFile的话则不会做这样的检查,比如上面的例子换成Assembly.LoadFile的话,则能正确载入版本2。

LoadFile:加载指定路径上的程序集文件的内容。LoadFrom: 根据程序集的文件名加载程序集文件的内容。

区别:

LoadFile 方法用来来加载和检查具有相同标识但位于不同路径中的程序集.但不会加载程序的依赖项。

LoadFrom 不能用于加载标识相同但路径不同的程序集。

 

------注意--------------

Assembly.Load(@"ReflectionTest");(项目需要添加这个dll的引用)

Assembly.LoadForm(@"C:/ReflectionTest.dll");

你可能感兴趣的:(C#反射之Assembly.Load,Assembly.LoadFile 与 Assembly.LoadFrom方法介绍)