C#中如何动态加载和卸载DLL

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C#中如何动态加载和卸载DLL相关的知识,希望对你有一定的参考价值。

在C#中我们也能使用Assembly.LoadFile实现动态加载DLL,但是当你试图卸载时,你会很惊讶的发现Assembly没有提供任何卸载的方法。这是由于托管代码的自动垃圾回收机制会做这件事情,所以C#不提供释放资源的函数,一切由垃圾回收来做。powered by 25175.net这引发了一个问题,用Assembly加载的DLL可能只在程序结束的时候才会被释放,这也意味着在程序运行期间无法更新被加载的DLL。而这个功能在某些程序设计时是非常必要的,考虑你正在用反射机制写一个查看DLL中所有函数详细信息的程序,程序提供一个菜单让用户可以选择DLL文件,这时就需要让程序能够卸载DLL,否则一旦用户重新得到新版本DLL时,必须要重新启动程序,重新选择加载DLL文件,这样的设计是用户无法忍受的。C#也提供了实现动态卸载DLL的方法,通过AppDomain来实现。AppDomain是一个独立执行应用程序的环境,当AppDomain被卸载的时候,在该环境中的所有资源也将被回收。关于AppDomain的详细资料参考MSDN。下面是使用AppDomain实现动态卸载DLL的代码,usingSystem;usingSystem.Collections.Generic;usingSystem.Text;usingSystem.Threading;usingSystem.Reflection;namespaceUnloadDllclassProgramstaticvoidMain(string[] args)stringcallingDomainName = AppDomain.CurrentDomain.FriendlyName;//Thread.GetDomain().FriendlyName;Console.WriteLine(callingDomainName);AppDomain ad = AppDomain.CreateDomain("DLL Unload test");ProxyObject obj = (ProxyObject)ad.CreateInstanceFromAndUnwrap(@"UnloadDll.exe","UnloadDll.ProxyObject");obj.LoadAssembly();obj.Invoke("TestDll.Class1","Test","It's a test");AppDomain.Unload(ad);obj =null;Console.ReadLine();classProxyObject : MarshalByRefObjectAssembly assembly =null;publicvoidLoadAssembly()assembly = Assembly.LoadFile(@"TestDLL.dll");publicboolInvoke(stringfullClassName,stringmethodName,paramsObject[] args)if(assembly ==null)returnfalse;Type tp = assembly.GetType(fullClassName);if(tp ==null)returnfalse;MethodInfo method = tp.GetMethod(methodName);if(method ==null)returnfalse;Object obj = Activator.CreateInstance(tp);method.Invoke(obj, args);returntrue;注意:1. 要想让一个对象能够穿过AppDomain边界,必须要继承MarshalByRefObject类,否则无法被其他AppDomain使用。 参考技术A 1、利用反射进行动态加载和调用.
Assembly assembly=Assembly.LoadFrom(DllPath); //利用dll的路径加载,同时将此程序集所依赖的程序集加载进来,需后辍名.dll
Assembly.LoadFile 只加载指定文件,并不会自动加载依赖程序集.Assmbly.Load无需后辍名
2、加载dll后,需要使用dll中某类.
Type type=ass.GetType(“TypeName”);//用类型的命名空间和名称获得类型
3、需要实例化类型,才可以使用,参数可以人为的指定,也可以无参数,静态实例可以省略
Object obj = Activator.CreateInstance(type,params[]);//利用指定的参数实例话类型
4、调用类型中的某个方法:
需要首先得到此方法
MethodInfo mi=type.GetMethod(“MehtodName”);//通过方法名称获得方法
5、然后对方法进行调用,多态性利用参数进行控制
mi.Invoke(obj,params[]);//根据参数直线方法,返回值就是原方法的返回值
参考技术B 难道不是自动的吗?

C#中动态加载和卸载DLL

  在C++中加载和卸载DLL是一件很容易的事,LoadLibrary和FreeLibrary让你能够轻易的在程序中加载DLL,然后在任何地方 卸载。在C#中我们也能使用Assembly.LoadFile实现动态加载DLL,但是当你试图卸载时,你会很惊讶的发现Assembly没有提供任何 卸载的方法。这是由于托管代码的自动垃圾回收机制会做这件事情,所以C#不提供释放资源的函数,一切由垃圾回收来做。 
  这引发了一个问题,用Assembly加载的DLL可能只在程序结束的时候才会被释放,这也意味着在程序运行期间无法更新被加载的DLL。而这个功能在某 些程序设计时是非常必要的,考虑你正在用反射机制写一个查看DLL中所有函数详细信息的程序,程序提供一个菜单让用户可以选择DLL文件,这时就需要让程 序能够卸载DLL,否则一旦用户重新得到新版本DLL时,必须要重新启动程序,重新选择加载DLL文件,这样的设计是用户无法忍受的。 
  C#也提供了实现动态卸载DLL的方法,通过AppDomain来实现。AppDomain是一个独立执行应用程序的环境,当AppDomain被卸载的 时候,在该环境中的所有资源也将被回收。关于AppDomain的详细资料参考MSDN。下面是使用AppDomain实现动态卸载DLL的代码:

 1 using System; 
 2 using System.Collections.Generic; 
 3 using System.Text; 
 4 using System.Threading; 
 5 using System.Reflection; 
 6 namespace UnloadDll 
 7 { 
 8 class Program 
 9 { 
10 static void Main(string[] args) 
11 { 
12 string callingDomainName = AppDomain.CurrentDomain.FriendlyName;//Thread.GetDomain().FriendlyName; 
13 Console.WriteLine(callingDomainName); 
14 AppDomain ad = AppDomain.CreateDomain("DLL Unload test"); 
15 ProxyObject obj = (ProxyObject)ad.CreateInstanceFromAndUnwrap(@"UnloadDll.exe", "UnloadDll.ProxyObject"); 
16 obj.LoadAssembly(); 
17 obj.Invoke("TestDll.Class1", "Test", "It‘s a test"); 
18 AppDomain.Unload(ad); 
19 obj = null; 
20 Console.ReadLine(); 
21 } 
22 } 
23 class ProxyObject : MarshalByRefObject 
24 { 
25 Assembly assembly = null; 
26 public void LoadAssembly() 
27 { 
28 assembly = Assembly.LoadFile(@"TestDLL.dll"); 
29 } 
30 public bool Invoke(string fullClassName, string methodName, params Object[] args) 
31 { 
32 if(assembly == null) 
33 return false; 
34 Type tp = assembly.GetType(fullClassName); 
35 if (tp == null) 
36 return false; 
37 MethodInfo method = tp.GetMethod(methodName); 
38 if (method == null) 
39 return false; 
40 Object obj = Activator.CreateInstance(tp); 
41 method.Invoke(obj, args); 
42 return true; 
43 } 
44 } 
45 } 

注意: 

1. 要想让一个对象能够穿过AppDomain边界,必须要继承MarshalByRefObject类,否则无法被其他AppDomain使用。 

2. 每个线程都有一个默认的AppDomain,可以通过Thread.GetDomain()来得到。

详细参考:http://www.cnblogs.com/foman/archive/2009/10/18/1585655.html

以上是关于C#中如何动态加载和卸载DLL的主要内容,如果未能解决你的问题,请参考以下文章

C#.Net 如何动态加载与卸载程序集(.dll或者.exe)

在 C# 中动态加载和使用 DLL

C#动态加载dll 时程序集的卸载问题

AppDomain 详解二-C#中动态加载和卸载DLL

vb.net编程,如何使用 appdomain 实现某进程DLL动态加载和卸载?

C# 写一个动态加载DLL的程序 怎么写?