最近幫底層開發的同時用C#重新封裝一下dll,也就是用C#類來封裝C++Dll里的方法,以供用戶使用。
之前也用到過類似的應用,大多數問題都出在類型轉換上,但是這次的應用層出不窮,所以在這里總結一下,以供自己以后查閱,也希望對大家能夠有所幫助。
首先,重復一下一些基本使用方法。具體的那些方式在這里就不重復講了,網上很多的。比如http://blog.csdn.net/sunboyljp/archive/2009/12/31/5110639.aspx
c++ 頭文件中的定義:
NPD_API int NP_Init();
C#中定義函數
[DllImport("npd_api.dll")]
public static extern int NP_Init();
基本類型轉換見下表(我用到過的):
BSTR——StringBuilder
LPCTSTR ——StringBuilder
LPCWSTR ——IntPtr
handle ——IntPtr
hwnd ——IntPtr
char * ——string
int * ——ref int
int & ——ref int
void * ——IntPtrs
unsigned char * ——ref byte
BOOL ——bool
DWORD ——uint或int(我用的是uint,沒出過什么問題)
我的問題來了,長期的經驗教訓我知道了:
1、指針做參數時在C#中一定要使用ref 或out關鍵字,尤其是結構體指針,要不會報內存讀取錯誤,即使不報錯數據也是不太對的。呵呵
SIPCLIENT_API void WINAPI SCCleanup(SipClient * psip);
[DllImport("sipclient.dll")]
public static extern void SCCleanup(ref SipClient psip);
其中SipClient是一個結構體。
2、重寫結構體的時候,之前有指明類型長度或數組長度的地方,也要進行相應的標注,要不也會導致內存錯誤。
代碼
typedef struct {
char sDVRIP[16]; /* DVR IP地址 */
char sDVRIPMask[16]; /* DVR IP地址掩碼 */
DWORD dwNetInterface; /* 10M/100M自適應,索引 */
WORD wDVRPort; /* 端口號 */
BYTE byMACAddr[MACADDR_LEN]; /* 服務器的物理地址 */
}NET_POSA_ETHERNET;
public struct NET_POSA_ETHERNET
{
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
public string sDVRIP; //DVR IP地址
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
public string sDVRIPMask; // DVR IP地址掩碼
public uint dwNetInterface; //網絡接口 1-10MBase-T 2-10MBase-T全雙工 3-100MBase-TX 4-100M全雙工 5-10M/100M自適應
public uint wDVRPort; //端口號
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]
public byte[] byMACAddr; //[MACADDR_LEN]; //PPPoE用戶名//服務器的物理地址
}
3、遇到這樣一個問題,折騰了大半天時間——http://space.cnblogs.com/q/16616/。
最后是在C++那邊做了修改解決的,通過制定模塊定義 (.def) 文件,統一制定導出函數對應的名稱。返回值為結構體指針的函數用IntPtr也能使用了。
代碼
SIPCLIENT_API SipClient* SCInit(const char * reaml, const char * from_ip, int from_port, const char * to_ip, int to_port, const char * server_id, const char * user_id, const char * user_name, void * user_obj_param);
[DllImport("sipclient.dll")]
public static extern IntPtr SCInit(string reaml, string from_ip, int from_port,
string to_ip, int to_port, string server_id,
string user_id, string user_name, IntPtr user_obj_param);
IntPtr client = IntPtr.Zero;
client = SIPCLIENT_API.SCInit(REALM, CLIENT_IP, CLIENT_PORT, SERVER_IP,
SERVER_PORT, SERVER_ID, USER_ID, USER_ID, IntPtr.Zero);
if (client != IntPtr.Zero)
sipclient = (SipClient)Marshal.PtrToStructure(client, typeof(SipClient));
else
MessageBox.Show("SipClient初始化失敗!");
4、后來還遇到個回調函數導致的崩潰問題,又耽誤了大半天時間,下班了還耽擱了會終於找的解決發辦法了。
剛開始同事分析出了崩潰的原因,都是回收方式惹的禍,可參見http://www.hudong.com/wiki/WINAPI,嘗試使用__stdcall,但是還是沒有解決問題
后來實踐證明,程序是很嚴謹的,半點差錯都不能出才不會導致錯誤,思路還是__stdcall,只不過少改了東西,有兩個地方需要改,才能保證不出錯。
參考http://hi.baidu.com/tease/blog/item/1fe7213802780f22b9998f5a.html。
關鍵就是這兩句話
typedef void (_stdcall *CiCiCallBack) (bool started, void* client,char *message);
將導出函數修改為:
extern "C" _declspec(dllexport) bool _stdcall Test(char* fileName, CiCiCallBack callback)
一開始的時候就只修改了定義那,卻忘記了導出時的修改,差點就放棄了這條解決思路了,不過還好,所謂堅持就是勝利!
5、后來封裝好拿到用戶那里用,卻總是提示說找不到C++那些dll.
網上一查,初步定位是開發環境引起的,跟環境部署有關系。我們的開發環境是vs2008,而客戶使用的vs2010,通過幾次嘗試,問題終於了。
首先考慮是缺少某些C++必備的運行庫,存在相互依賴關系,所以導致找不到dll。用查看Dependency Walker查看才發現真的是客戶機子上少了一些東西。
但是此路不通,將缺少的那些東西拷貝到可執行程序目錄下,問題依舊沒有解決。但是依舊堅持這條路~
嘗試安裝vcredist_x86.exe,以排除是否還是缺少了某些運行庫的可能,問題依然存在。
后來我想起來之前搜索問題的時候,看到好像跟dll的Releas\Debug版本還有關系,所有又嘗試提議讓同事將他們的c++dll改為Release版的。
因為項目是多個人一起做了,編譯Release版還花了不少時間,不過好歹問題終於解決了!
總結:直接安裝vcredist_x86.exe,所有dll必須使用Release版的。如果使用Debug版的就必須保證可執行程序目錄下的dll是完整的,缺一不可!
網上詳細的講解也很多,感覺這個總結的很好http://hi.baidu.com/fairysky/blog/item/e7a8366dbaa735f3431694c8.html。
做程序就怕出現問題,出現問題就怕不知道原因,知道原因了就好找解決的辦法啦!