原文:
c# 動態加載和卸載DLL程序集
在 C++中加載和卸載DLL是一件很容易的事,LoadLibrary和FreeLibrary讓你可以輕易的在程序中加載DLL,而後在任何地方卸載。在 C#中咱們也能使用Assembly.LoadFile實現動態加載DLL,可是當你試圖卸載時,你會很驚訝的發現Assembly沒有提供任何卸載的方 法。這是因爲託管代碼的自動垃圾回收機制會作這件事情,因此C#不提供釋放資源的函數,一切由垃圾回收來作。
當AppDomain被卸載的時候,在該環境中的全部資源也將被回收。關於AppDomain的詳細資料參考MSDN。下面是使用AppDomain實現動態卸載DLL的代碼,
namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
private void button1_Click(object sender, EventArgs e)
{
string callingDomainName = AppDomain.CurrentDomain.FriendlyName;
AppDomain ad = AppDomain.CreateDomain("DLL Add test");
ProxyObject obj = (ProxyObject)ad.CreateInstanceFromAndUnwrap(@"WindowsFormsApplication1.exe", "WindowsFormsApplication1.ProxyObject");
obj.LoadAssembly();
String res = obj.Invoke("App.TestFrm", "Add", Convert.ToInt32 ( textBox1.Text) ,Convert.ToInt32 (textBox2 .Text) );
textBox3.Text = res;
AppDomain.Unload(ad);
obj = null;
}
}
class ProxyObject : MarshalByRefObject
{
Assembly assembly = null;
public void LoadAssembly()
{
assembly = Assembly.LoadFile(@"D:\Documents\Work\Dev\項目\應用程序域\應用程序域\WindowsFormsApplication1\bin\Debug\App.dll");
}
public String Invoke(string fullClassName, string methodName, params Object[] args)
{
if (assembly == null)
return "";
Type tp = assembly.GetType(fullClassName);
if (tp == null)
return "";
MethodInfo method = tp.GetMethod(methodName);
if (method == null)
return "";
Object obj = Activator.CreateInstance(tp);
String res = Convert.ToString ( method.Invoke(obj, args)) ;
return res ;
}
}
}
注意:
1. 要想讓一個對象可以穿過AppDomain邊界,必需要繼承MarshalByRefObject類,不然沒法被其餘AppDomain使用。
2. 每一個線程都有一個默認的AppDomain,能夠經過Thread.GetDomain()來獲得