博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
C#中动态加载和卸载DLL
阅读量:5166 次
发布时间:2019-06-13

本文共 2226 字,大约阅读时间需要 7 分钟。

 

原文:

在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的代码,

using System;using System.Collections.Generic;using System.Text;using System.Threading;using System.Reflection;namespace UnloadDll{    class Program    {        static void Main(string[] args)        {            string callingDomainName = 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();        }    }    class ProxyObject : MarshalByRefObject    {        Assembly assembly = null;        public void LoadAssembly()        {            assembly = Assembly.LoadFile(@"TestDLL.dll");        }        public bool Invoke(string fullClassName, string methodName, params Object[] args)        {            if (assembly == null)                return false;            Type tp = assembly.GetType(fullClassName);            if (tp == null)                return false;            MethodInfo method = tp.GetMethod(methodName);            if (method == null)                return false;            Object obj = Activator.CreateInstance(tp);            method.Invoke(obj, args);            return true;        }    }}

 

 

注意:

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

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

转载:地址不详^_^

posted on
2017-12-03 23:54 阅读(
...) 评论(
...)

转载于:https://www.cnblogs.com/lonelyxmas/p/7968041.html

你可能感兴趣的文章
dubbo序列化hibernate.LazyInitializationException could not initialize proxy - no Session懒加载异常的解决...
查看>>
jQuery中的事件绑定的几种方式
查看>>
泥塑课
查看>>
iOS 自定义的对象类型的解档和归档
查看>>
setImageBitmap和setImageResource
查看>>
springMVC4 注解配置实例
查看>>
单片机编程
查看>>
Filter in Servlet
查看>>
Linux--SquashFS
查看>>
Application Pool Identities
查看>>
2017-3-24 开通博客园
查看>>
【MySQL性能优化】MySQL常见SQL错误用法
查看>>
3.6 字符串
查看>>
Vue2全家桶之一:vue-cli(vue脚手架)超详细教程
查看>>
Struts 2 常用技术
查看>>
树形DP
查看>>
Springboot实现上传文件接口,使用python的requests进行组装报文上传文件的方法
查看>>
python flask解决上传下载的问题
查看>>
博客园原始直链视频插入
查看>>
语法测试
查看>>