如何在C#中加載自己編寫(xiě)的動(dòng)態(tài)鏈接庫(DLL)
李偉華 msn:liweihua200204@hotmail.com
摘要 本文主要講述如何在C#中逐步實(shí)現加載自己用C++語(yǔ)言編寫(xiě)的動(dòng)態(tài)鏈接庫,以及在導入時(shí)如何進(jìn)行C#和C++語(yǔ)言的數據類(lèi)型匹配
關(guān)鍵詞 C# C++ 動(dòng)態(tài)鏈接庫 加載 數據類(lèi)型匹配
一、發(fā)生的背景
在開(kāi)發(fā)新項目中使用了新的語(yǔ)言開(kāi)發(fā)C#和新的技術(shù)方案WEB Service,但是在新項目中,一些舊的模塊需要繼續使用,一般是采用C或C++或Delphi編寫(xiě)的,如何利用舊模塊對于開(kāi)發(fā)人員來(lái)說(shuō),有三種可用方法供選擇:第一、將C或C++函數用C#徹底改寫(xiě)一遍,這樣整個(gè)項目代碼比較統一,維護也方便一些。但是盡管微軟以及某些書(shū)籍說(shuō),C#和C++如何接近,但是改寫(xiě)起來(lái)還是很痛苦的事情,特別是C++里的指針和內存操作;第二、將C或C++函數封裝成COM,在C#中調用COM比較方便,只是在封裝時(shí)需要處理C或C++類(lèi)型和COM類(lèi)型之間的轉換,也有一些麻煩,另外COM還需要注冊,注冊次數多了又可能導致混亂;第三、將C或C++函數封裝成動(dòng)態(tài)鏈接庫,封裝的過(guò)程簡(jiǎn)單,工作量不大。因此我決定采用加載動(dòng)態(tài)鏈接庫的方法實(shí)現,于是產(chǎn)生了在C#中如何調用自定義的動(dòng)態(tài)鏈接庫問(wèn)題,我在網(wǎng)上搜索相關(guān)主題,發(fā)現一篇調用系統API的文章,但是沒(méi)有說(shuō)明如何解決此問(wèn)題,在MSDN上也沒(méi)有相關(guān)詳細說(shuō)明?;诖?,我決定自己從簡(jiǎn)單出發(fā),逐步試驗,看看能否達到自己的目標。
(說(shuō)明一點(diǎn):我這里改寫(xiě)為什么很怕麻煩,我改寫(xiě)的代碼是變長(cháng)加密算法函數,代碼有600多行,對算法本身不熟悉,算法中指針和內存操作太多,要想保證算法正確,最可行的方法就是少動(dòng)代碼,否則只要有一點(diǎn)點(diǎn)差錯,就不能肯定算法與以前兼容)
二、技術(shù)實(shí)現
下面看看如何逐步實(shí)現動(dòng)態(tài)庫的加載,類(lèi)型的匹配:
動(dòng)態(tài)鏈接庫函數導出的定義,這個(gè)不需要多說(shuō),大家參考下面宏定義即可:
#define LIBEXPORT_API extern "C" __declspec(dllexport)
第一步,我先從簡(jiǎn)單的調用出發(fā),定義了一個(gè)簡(jiǎn)單的函數,該函數僅僅實(shí)現一個(gè)整數加法求和:
LIBEXPORT_API int mySum(int a,int b){ return a+b;}
C#定義導入定義:
public class RefComm
{
[DllImport("LibEncrypt.dll", EntryPoint=" mySum ",CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)] public static extern int mySum (int a,int b);
}
在C#中調用測試:
int iSum= RefComm. mySum(2,3);
運行查看結果iSum為5,調用正確。第一步試驗完成,說(shuō)明在C#中能夠調用自定義的動(dòng)態(tài)鏈接庫函數。
第二步,我定義了字符串操作的函數(簡(jiǎn)單起見(jiàn),還是采用前面的函數名),返回結果為字符串:
LIBEXPORT_API char *mySum(char *a,char *b){sprintf(b,”%s”,a) return a;}
C#定義導入定義:
public class RefComm
{
[DllImport("LibEncrypt.dll", EntryPoint=" mySum ",CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, string b);
}
在C#中調用測試:
string strDest=””;
string strTmp= RefComm. mySum(“12345”, strDest);
運行查看結果strTmp為“12345”,但是strDest為空。
我修改動(dòng)態(tài)鏈接庫實(shí)現,返回結果為串b:
LIBEXPORT_API char *mySum(char *a,char *b){sprintf(b,”%s”,a) return b;}
修改C#導入定義,將串b修改為ref方式:
public class RefComm
{
[DllImport("LibEncrypt.dll", EntryPoint=" mySum ",CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, ref string b);
}
在C#中再調用測試:
string strDest=””;
string strTmp= RefComm. mySum(“12345”, ref strDest);
運行查看結果strTmp和strDest均不對,含不可見(jiàn)字符。
再修改C#導入定義,將CharSet從Auto修改為Ansi:
public class RefComm
{
[DllImport("LibEncrypt.dll", EntryPoint=" mySum ",CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, string b);
}
在C#中再調用測試:
string strDest=””;
string strTmp= RefComm. mySum(“12345”, ref strDest);
運行查看結果strTmp為“12345”,但是串strDest沒(méi)有賦值。第二步實(shí)現函數返回串,但是在函數出口參數中沒(méi)能進(jìn)行輸出。
再次修改C#導入定義,將串b修改為引用(ref):
public class RefComm
{
[DllImport("LibEncrypt.dll", EntryPoint=" mySum ",CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, ref string b);
}
運行時(shí)調用失敗,不能繼續執行。
第三步,修改動(dòng)態(tài)鏈接庫實(shí)現,將b修改為雙重指針:
LIBEXPORT_API char *mySum(char *a,char **b){sprintf((*b),”%s”,a) return *b;}
C#導入定義:
public class RefComm
{
[DllImport("LibEncrypt.dll", EntryPoint=" mySum ",CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)] public static extern string mySum (string a, ref string b);
}
在C#中調用測試:
string strDest=””;
string strTmp= RefComm. mySum(“12345”, ref strDest);
運行查看結果strTmp和strDest均為“12345”,調用正確。第三步實(shí)現了函數出口參數正確輸出結果。
第四步,修改動(dòng)態(tài)鏈接庫實(shí)現,實(shí)現整數參數的輸出:
LIBEXPORT_API int mySum(int a,int b,int *c){ *c=a+b; return *c;}
C#導入的定義:
public class RefComm
{
[DllImport("LibEncrypt.dll", EntryPoint=" mySum ",CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)] public static extern int mySum (int a, int b,ref int c);
}
在C#中調用測試:
int c=0;
int iSum= RefComm. mySum(2,3, ref c);
運行查看結果iSum 和c均為5,調用正確。
經(jīng)過(guò)以上幾個(gè)步驟的試驗,基本掌握了如何定義動(dòng)態(tài)庫函數以及如何在C#定義導入,有此基礎,很快我實(shí)現了變長(cháng)加密函數在C#中的調用,至此目標實(shí)現。
三、結論
在C#中,調用C++編寫(xiě)動(dòng)態(tài)鏈接庫函數,如果需要出口參數輸出,則需要使用指針,對于字符串,則需要使用雙重指針,對于C#的導入定義,則需要使用引用(ref)定義。
對于函數返回值,C#導入定義和C++動(dòng)態(tài)庫函數申明定義需要保持一致,否則會(huì )出現函數調用失敗。
定義導入時(shí),一定注意CharSet和CallingConvention參數,否則導致調用失敗或結果異常。
運行時(shí),動(dòng)態(tài)鏈接庫放在C#程序的目錄下即可,我這里是一個(gè)C#的動(dòng)態(tài)鏈接庫,兩個(gè)動(dòng)態(tài)鏈接庫就在同一個(gè)目錄下運行。
Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=259213
聯(lián)系客服