1. 程式人生 > >使用AppDomain進行動態加載和卸載dll

使用AppDomain進行動態加載和卸載dll

加載 rec file cache brush 實例 var 被占用 void

加載dll最簡單的寫法

Assembly.Load("<dll路徑>")

  

但這樣寫這個dll就被程序占用不可刪除,即不可以卸載。

通過AppDomain加代理的方式進行加載則可以實現卸載,從而實現程序不被占用並被刪除。

第一步,創建AssemblyProxy類

public class AssemblyProxy : MarshalByRefObject
{
        private Assembly assembly;

        public void LoadAssembly(string fullName)
        {
            assembly 
= Assembly.LoadFrom(fullName); }/// <summary> /// 開放委托,可使用Assembly操作 /// </summary> /// <param name="action"></param> public T Work<T>(Func<Assembly, T> action) { return action.Invoke(assembly); } }

AssemblyProxy類將運行在子AppDomain中,通過操作AssemblyProxy類進行反射加載的DLL。保證AppDomain.CurrentDomain和子AppDomain隔離,最後卸載子AppDomain就實現了動態加載和卸載dll的目標。

第二步,實現AssemblyLoader類

public class AssemblyLoader
{
        private AppDomain appDomain;

        public AssemblyProxy Proxy;

        public AssemblyLoader()
        {
            AppDomainSetup setup 
= new AppDomainSetup(); setup.ApplicationName = "Child"; setup.ApplicationBase = AppDomain.CurrentDomain.BaseDirectory; setup.PrivateBinPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "private"); setup.CachePath = setup.ApplicationBase; setup.ShadowCopyFiles = "true"; setup.ShadowCopyDirectories = setup.ApplicationBase; appDomain = AppDomain.CreateDomain("ChildDomain", null, setup); string name = Assembly.GetExecutingAssembly().GetName().FullName;
       // 從子類AppDomain中反射創建AssemblyProxys實例 Proxy
= (AssemblyProxy)appDomain.CreateInstanceAndUnwrap(name, typeof(AssemblyProxy).FullName); } public void LoadAssembly(string fullName) { Proxy.LoadAssembly(fullName); } public void Unload() { AppDomain.Unload(appDomain); appDomain = null; } }

接著就可以使用了

var loader = new AssemblyLoader();
loader.LoadAssembly(path);

var version = loader.Proxy.Work(assembly =>
{
     return assembly.GetName().Version.ToString();
});

loader.Unload();

使用AppDomain進行動態加載和卸載dll