最近繼續用ASP.NET來重新開發ACM的Online Judge系統,因為要進行進程的監控,所以自己編寫了一個非託管的DLL供ASP.NET調用。
我用的是VS2005的開發環境,後來發現使用[DllImport("Judge.dll")]後提示 無法載入 DLL “Judge.dll” 找不到指定的模組
我這時是把Judge.dll拷貝到Bin目錄下的,但仍然提示找不到DLL,在工程裡添加DLL引用的時候,發現添加這個非託管DLL就會令VS2005異常退出(上網搜尋後也發現有人有相同的問題)
後來發現用[DllImport(@"C:\OJ\Bin\Judge.dll")]
這樣指定DLL的絕對路徑就可以正常裝載。
這裡還有一個解決辦法http://forums.asp.net/thread/1121085.aspx
這個問題最常出現在使用第三方非託管DLL組件的時候,我的也同樣是這時出的問題,Asp.Net Team的官方解決方案如下:
首先需要確認你引用了哪些組件,那些是託管的,哪些是非託管的.託管的很好辦,直接被使用的需要引用,間接使用的需要拷貝到bin目錄下.非託管的處理會比較麻煩.實際上,你拷貝到bin沒有任何協助,因為CLR會把檔案拷貝到一個臨時目錄下,然後在那運行web,而CLR只會拷貝託管檔案,這就是為什麼我們明明把非託管的dll放在了bin下卻依然提示不能載入模組了.
具體做法如下:
首先我們在伺服器上隨便找個地方建立一個目錄,假如為C:\DLL
然後,在環境變數中,給Path變數添加這個目錄
最後,把所有的非託管檔案都拷貝到C:\DLL中.
或者更乾脆的把DLL放到system32目錄
對於可以自己部署的應用程式,這樣未償不是一個解決辦法,然而,如果我們用的是虛擬空間,我們是沒辦法把註冊PATH變數或者把我們自己的DLL拷到system32目錄的。同時我們也不一定知道我們的Dll的物理路徑
DllImport裡面只能用字串常量,而不能夠用Server.MapPath(@"~/Bin/Judge.dll")來確定物理路徑。
經過一翻研究,終於想到了一個完美的解決辦法
首先我們用
[DllImport("kernel32.dll")]
private extern static IntPtr LoadLibrary(String path);
[DllImport("kernel32.dll")]
private extern static IntPtr GetProcAddress(IntPtr lib, String funcName);
[DllImport("kernel32.dll")]
private extern static bool FreeLibrary(IntPtr lib);
分別取得了LoadLibrary和GetProcAddress函數的地址,再通過這兩個函數來取得我們的DLL裡面的函數。
我們可以先用Server.MapPath(@"~/Bin/Judge.dll")來取得我們的DLL的物理路徑,然後再用LoadLibrary進行載入,最後用GetProcAddress取得要用的函數地址
以下自訂類的程式碼完成LoadLibrary的裝載和函數調用:
public class DllInvoke
{
[DllImport("kernel32.dll")]
private extern static IntPtr LoadLibrary(String path);
[DllImport("kernel32.dll")]
private extern static IntPtr GetProcAddress(IntPtr lib, String funcName);
[DllImport("kernel32.dll")]
private extern static bool FreeLibrary(IntPtr lib);
private IntPtr hLib;
public DllInvoke(String DLLPath)
{
hLib = LoadLibrary(DLLPath);
}
~DllInvoke()
{
FreeLibrary(hLib);
}
//將要執行的函數轉換為委託
public Delegate Invoke(String APIName,Type t)
{
IntPtr api = GetProcAddress(hLib, APIName);
return (Delegate)Marshal.GetDelegateForFunctionPointer(api,t);
}
}
下面代碼進行調用
public delegate int Compile(String command, StringBuilder inf);//編譯
DllInvoke dll = new DllInvoke(Server.MapPath(@"~/Bin/Judge.dll"));
Compile compile = (Compile)dll.Invoke("Compile", typeof(Compile));
StringBuilder inf;
compile(@“gcc a.c -o a.exe“,inf); //這裡就是調用我的DLL裡定義的Compile函數