c#調用com的總結貼

c# 無法加載DLL“###.dll”,: 找不到指定的模塊。 (異常來自 HRESULT:0x8007007E)

3830人閱讀 評論(3) 收藏 舉報

    http://www.itwis.com/html/net/net20/20071008/366.html

我用的是VS2005的開發環境,後來發現使用[DllImport("Judge.dll")]後提示 無法加載 DLL “Judge.dll” 找不到指定的模塊!我這時就把Judge.dll拷貝到Bin目錄下,但仍然提示找不到DLL,在工程裏添加DLL引用的時候,發現添加這個非託管DLL就會令VS2005異常退出(上網搜索後也發現有人有相同的問題)。

  後來發現用[DllImport(@"C:/OJ/Bin/Judge.dll")]這樣指定DLL的絕對路徑就可以正常裝載。

  這裏還有一個解決辦法.Net/thread/1121085.aspx">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函數

 

 

http://blog.csdn.net/jslfb2008/archive/2010/04/08/5460319.aspx

 

如果是非託管的,就用DllImport,舉例  
  using System;  
  using System.Runtime.InteropServices;  
  class MainApp  
  [DllImport("Kernel32")] //讀取動態庫文件  
  public static extern int GetProcAddress(int handle, String funcname);  

  給你講一下我的經驗:  
  首先 你在C#中調用的 是C++ 寫的一個動態庫。比如Kernel32.dll 中的 函數;  
  這個函數用C++寫 有如下要求:  
1、 必須爲全局函數  
2、 函數參數 必須爲基本類型,也就是C++ 和C#都有的類型,否則你在public static extern int GetProcAddress(int handle, String funcname);  
  這裏沒有辦法聲明。 其餘的 沒什麼了;  

先前用VC寫的程序生成的DLL,C#裏能夠調用麼?  
可以的話,在VC裏傳入參數爲CString,而C#裏是string,怎麼傳參數呢?  

再者就是問,用VC寫的DLL如果是一個導出類,而不是函數接口,C#裏可以直接調用麼?  
第一個問題我遇到過, 可用如下方法解決:  

  VC++ 中爲: int Set(char** str,int n); //將 String 改成 char** , C# 中沒有與String對應的類型  
  C# 中爲: int Set(ref string str,int n);  
   
  VC++ 中的 BOOL 類型對應 C# 中的 System.Int32 類型, 建議改動一下參數類型吧.

第二個問題,生成的DLL裏的函數在一個類裏面,這樣的話在C#裏需要實例化那個類麼?怎麼做  
比如說,類solution裏有函數int getch(int a);  
我怎麼調用這個getch函數??  

建議在C++中另外寫個函數封裝一下, 如:  
  int Dllgetch(int a)  
{  
  solution st = //實例化 類solution  
  return st.getch(a);  
}  
這個 Dllgetch(int a) 就可以提取出來供 C# 調用.
你的方法可以,我也解決了  
將CString 改成了LPCTSTR  
BOOL在C#中對應的是Boolean

****************************上段摘自jingshuaizh回覆提問者的答案,講得很詳細**********************

還有你可以參考這裏:http://blog.csdn.net/jingshuaizh/archive/2009/02/04/3862019.aspx

DLL文件放置位置:

(1)項目文件夾下的../bin/Debug/

(2)系統文件夾下的C:/WINDOWS/system32

如果能確定位置是正確的,那麼就得看看是不是你調用的DLL文件是不是使用到其他DLL文件裏的內容,這點很關鍵,在找不到其他問題的情況下這種情況的可能性高過百分之九十,這只是我遇到的情況,要是有其他情況,可以發來一起討論。。。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章