最近繼續(xù)用ASP.Net來(lái)重新開發(fā)ACM的Online Judge系統(tǒng),因?yàn)橐M(jìn)行進(jìn)程的監(jiān)控,所以自己編寫了一個(gè)非托管的DLL供ASP.Net調(diào)用。
我用的是VS2005的開發(fā)環(huán)境,后來(lái)發(fā)現(xiàn)使用[DllImport("Judge.dll")]后提示 無(wú)法加載 DLL “Judge.dll” 找不到指定的模塊!我這時(shí)就把Judge.dll拷貝到Bin目錄下,但仍然提示找不到DLL,在工程里添加DLL引用的時(shí)候,發(fā)現(xiàn)添加這個(gè)非托管DLL就會(huì)令VS2005異常退出(上網(wǎng)搜索后也發(fā)現(xiàn)有人有相同的問題)。
后來(lái)發(fā)現(xiàn)用[DllImport(@"C:\OJ\Bin\Judge.dll")]這樣指定DLL的絕對(duì)路徑就可以正常裝載。
這里還有一個(gè)解決辦法.http://forums.asp.Net/thread/1121085.aspx
這個(gè)問題最常出現(xiàn)在使用第三方非托管DLL組件的時(shí)候,我的也同樣是這時(shí)出的問題,Asp.Net Team的官方解決方案如下:
首先需要確認(rèn)你引用了哪些組件,那些是托管的,哪些是非托管的.托管的很好辦,直接被使用的需要引用,間接使用的需要拷貝到bin目錄下.非托管的處理會(huì)比較麻煩.實(shí)際上,你拷貝到bin沒有任何幫助,因?yàn)镃LR會(huì)把文件拷貝到一個(gè)臨時(shí)目錄下,然后在那運(yùn)行web,而CLR只會(huì)拷貝托管文件,這就是為什么我們明明把非托管的dll放在了bin下卻依然提示不能加載模塊了.
具體做法如下:
首先我們?cè)诜⻊?wù)器上隨便找個(gè)地方新建一個(gè)目錄,假如為C:\DLL
然后,在環(huán)境變量中,給Path變量添加這個(gè)目錄
最后,把所有的非托管文件都拷貝到C:\DLL中.
或者更干脆的把DLL放到system32目錄
對(duì)于可以自己部署的應(yīng)用程序,這樣未償不是一個(gè)解決辦法,然而,如果我們用的是虛擬空間,我們是沒辦法把注冊(cè)PATH變量或者把我們自己的DLL拷到system32目錄的。同時(shí)我們也不一定知道我們的Dll的物理路徑。
DllImport里面只能用字符串常量,而不能夠用Server.MapPath(@"~/Bin/Judge.dll")來(lái)確定物理路徑。
經(jīng)過(guò)一翻研究,終于想到了一個(gè)完美的解決辦法。
首先我們用
[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函數(shù)的地址,再通過(guò)這兩個(gè)函數(shù)來(lái)取得我們的DLL里面的函數(shù)。
我們可以先用Server.MapPath(@"~/Bin/Judge.dll")來(lái)取得我們的DLL的物理路徑,然后再用LoadLibrary進(jìn)行載入,最后用GetProcAddress取得要用的函數(shù)地址。
以下是自定義類的代碼完成LoadLibrary的裝載和函數(shù)調(diào)用:
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); } //將要執(zhí)行的函數(shù)轉(zhuǎn)換為委托 public Delegate Invoke(String APIName,Type t) { IntPtr api = GetProcAddress(hLib, APIName); return (Delegate)Marshal.GetDelegateForFunctionPointer(api,t); } } 用下面代碼進(jìn)行調(diào)用
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); //這里就是調(diào)用我的DLL里定義的Compile函數(shù)
|