使用AppDomain進行動態加載和卸載dll
阿新 • • 發佈:2019-03-18
加載 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