摘要 本文主要講述如何在c#中逐步實(shí)現(xiàn)加載自己用c++語言編寫的動態(tài)鏈接庫,以及在導(dǎo)入時如何進(jìn)行c#和c++語言的數(shù)據(jù)類型匹配
關(guān)鍵詞 c# c++ 動態(tài)鏈接庫 加載 數(shù)據(jù)類型匹配
一、發(fā)生的背景
在開發(fā)新項(xiàng)目中使用了新的語言開發(fā)c#和新的技術(shù)方案web service,但是在新項(xiàng)目中,一些舊的模塊需要繼續(xù)使用,一般是采用c或c++或delphi編寫的,如何利用舊模塊對于開發(fā)人員來說,有三種可用方法供選擇:第一、將c或c++函數(shù)用c#徹底改寫一遍,這樣整個項(xiàng)目代碼比較統(tǒng)一,維護(hù)也方便一些。但是盡管微軟以及某些書籍說,c#和c++如何接近,但是改寫起來還是很痛苦的事情,特別是c++里的指針和內(nèi)存操作;第二、將c或c++函數(shù)封裝成com,在c#中調(diào)用com比較方便,只是在封裝時需要處理c或c++類型和com類型之間的轉(zhuǎn)換,也有一些麻煩,另外com還需要注冊,注冊次數(shù)多了又可能導(dǎo)致混亂;第三、將c或c++函數(shù)封裝成動態(tài)鏈接庫,封裝的過程簡單,工作量不大。因此我決定采用加載動態(tài)鏈接庫的方法實(shí)現(xiàn),于是產(chǎn)生了在c#中如何調(diào)用自定義的動態(tài)鏈接庫問題,我在網(wǎng)上搜索相關(guān)主題,發(fā)現(xiàn)一篇調(diào)用系統(tǒng)api的文章,但是沒有說明如何解決此問題,在msdn上也沒有相關(guān)詳細(xì)說明。基于此,我決定自己從簡單出發(fā),逐步試驗(yàn),看看能否達(dá)到自己的目標(biāo)。
(說明一點(diǎn):我這里改寫為什么很怕麻煩,我改寫的代碼是變長加密算法函數(shù),代碼有600多行,對算法本身不熟悉,算法中指針和內(nèi)存操作太多,要想保證算法正確,最可行的方法就是少動代碼,否則只要有一點(diǎn)點(diǎn)差錯,就不能肯定算法與以前兼容)
二、技術(shù)實(shí)現(xiàn)
下面看看如何逐步實(shí)現(xiàn)動態(tài)庫的加載,類型的匹配:
動態(tài)鏈接庫函數(shù)導(dǎo)出的定義,這個不需要多說,大家參考下面宏定義即可:
#define libexport_api extern "c" __declspec(dllexport)
第一步,我先從簡單的調(diào)用出發(fā),定義了一個簡單的函數(shù),該函數(shù)僅僅實(shí)現(xiàn)一個整數(shù)加法求和:
libexport_api int mysum(int a,int b){ return a+b;}
c#定義導(dǎo)入定義:
public class refcomm
{
[dllimport("libencrypt.dll", entrypoint=" mysum ",charset=charset.auto,callingconvention=callingconvention.stdcall)] public static extern int mysum (int a,int b);
}
在c#中調(diào)用測試:
int isum= refcomm. mysum(2,3);
運(yùn)行查看結(jié)果isum為5,調(diào)用正確。第一步試驗(yàn)完成,說明在c#中能夠調(diào)用自定義的動態(tài)鏈接庫函數(shù)。
第二步,我定義了字符串操作的函數(shù)(簡單起見,還是采用前面的函數(shù)名),返回結(jié)果為字符串:
libexport_api char *mysum(char *a,char *b){sprintf(b,”%s”,a) return a;}
c#定義導(dǎo)入定義:
public class refcomm
{
[dllimport("libencrypt.dll", entrypoint=" mysum ",charset=charset.auto,callingconvention=callingconvention.stdcall)] public static extern string mysum (string a, string b);
}
在c#中調(diào)用測試:
string strdest=””;
string strtmp= refcomm. mysum(“12345”, strdest);
運(yùn)行查看結(jié)果strtmp為“12345”,但是strdest為空。
我修改動態(tài)鏈接庫實(shí)現(xiàn),返回結(jié)果為串b:
libexport_api char *mysum(char *a,char *b){sprintf(b,”%s”,a) return b;}
修改c#導(dǎo)入定義,將串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#中再調(diào)用測試:
string strdest=””;
string strtmp= refcomm. mysum(“12345”, ref strdest);
運(yùn)行查看結(jié)果strtmp和strdest均不對,含不可見字符。
再修改c#導(dǎo)入定義,將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#中再調(diào)用測試:
string strdest=””;
string strtmp= refcomm. mysum(“12345”, ref strdest);
運(yùn)行查看結(jié)果strtmp為“12345”,但是串strdest沒有賦值。第二步實(shí)現(xiàn)函數(shù)返回串,但是在函數(shù)出口參數(shù)中沒能進(jìn)行輸出。
再次修改c#導(dǎo)入定義,將串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);
}
運(yùn)行時調(diào)用失敗,不能繼續(xù)執(zhí)行。
第三步,修改動態(tài)鏈接庫實(shí)現(xiàn),將b修改為雙重指針:
libexport_api char *mysum(char *a,char **b){sprintf((*b),”%s”,a) return *b;}
c#導(dǎo)入定義:
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#中調(diào)用測試:
string strdest=””;
string strtmp= refcomm. mysum(“12345”, ref strdest);
運(yùn)行查看結(jié)果strtmp和strdest均為“12345”,調(diào)用正確。第三步實(shí)現(xiàn)了函數(shù)出口參數(shù)正確輸出結(jié)果。
第四步,修改動態(tài)鏈接庫實(shí)現(xiàn),實(shí)現(xiàn)整數(shù)參數(shù)的輸出:
libexport_api int mysum(int a,int b,int *c){ *c=a+b; return *c;}
c#導(dǎo)入的定義:
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#中調(diào)用測試:
int c=0;
int isum= refcomm. mysum(2,3, ref c);
運(yùn)行查看結(jié)果isum 和c均為5,調(diào)用正確。
經(jīng)過以上幾個步驟的試驗(yàn),基本掌握了如何定義動態(tài)庫函數(shù)以及如何在c#定義導(dǎo)入,有此基礎(chǔ),很快我實(shí)現(xiàn)了變長加密函數(shù)在c#中的調(diào)用,至此目標(biāo)實(shí)現(xiàn)。
三、結(jié)論
在c#中,調(diào)用c++編寫動態(tài)鏈接庫函數(shù),如果需要出口參數(shù)輸出,則需要使用指針,對于字符串,則需要使用雙重指針,對于c#的導(dǎo)入定義,則需要使用引用(ref)定義。
對于函數(shù)返回值,c#導(dǎo)入定義和c++動態(tài)庫函數(shù)申明定義需要保持一致,否則會出現(xiàn)函數(shù)調(diào)用失敗。
定義導(dǎo)入時,一定注意charset和callingconvention參數(shù),否則導(dǎo)致調(diào)用失敗或結(jié)果異常。
運(yùn)行時,動態(tài)鏈接庫放在c#程序的目錄下即可,我這里是一個c#的動態(tài)鏈接庫,兩個動態(tài)鏈接庫就在同一個目錄下運(yùn)行。