如何在C 中載入自己編寫的動態鏈結庫(DLL)

2021-03-31 08:57:00 字數 4004 閱讀 1012

如何在c#中載入自己編寫的動態鏈結庫(dll) 

李偉華 msn:liweihua200204@hotmail.***

一、發生的背景

在開發新專案中使用了新的語言開發c#和新的技術方案web service,但是在新專案中,一些舊的模組需要繼續使用,一般是採用c或c++或delphi編寫的,如何利用舊模組對於開發人員來說,有三種可用方法供選擇:第

一、將c或c++函式用c#徹底改寫一遍,這樣整個專案**比較統一,維護也方便一些。但是儘管微軟以及某些書籍說,c#和c++如何接近,但是改寫起來還是很痛苦的事情,特別是c++裡的指標和記憶體操作;第

二、將c或c++函式封裝成***,在c#中呼叫***比較方便,只是在封裝時需要處理c或c++型別和***型別之間的轉換,也有一些麻煩,另外***還需要註冊,註冊次數多了又可能導致混亂;第

三、將c或c++函式封裝成動態鏈結庫,封裝的過程簡單,工作量不大。因此我決定採用載入動態鏈結庫的方法實現,於是產生了在c#中如何呼叫自定義的動態鏈結庫問題,我在網上搜尋相關主題,發現一篇呼叫系統api的文章,但是沒有說明如何解決此問題,在msdn上也沒有相關詳細說明。基於此,我決定自己從簡單出發,逐步試驗,看看能否達到自己的目標。

(說明一點:我這裡改寫為什麼很怕麻煩,我改寫的**是變長加密演算法函式,**有600多行,對演算法本身不熟悉,演算法中指標和記憶體操作太多,要想保證演算法正確,最可行的方法就是少動**,否則只要有一點點差錯,就不能肯定演算法與以前相容)

二、技術實現

下面看看如何逐步實現動態庫的載入,型別的匹配:

#define libexport_api extern "c" __declspec(dllexport)

第一步,我先從簡單的呼叫出發,定義了乙個簡單的函式,該函式僅僅實現乙個整數加法求和:

libexport_api int mysum(int a,int b)

c#定義匯入定義:

public class ref***m

[dllimport("libencrypt.dll", entrypoint=" mysum ",charset=charset.auto,callingconvention=callingconvention.stdcall)] public static extern int mysum (int a,int b);

在c#中呼叫測試:

int isum= ref***m. mysum(2,3);

執行檢視結果isum為5,呼叫正確。第一步試驗完成,說明在c#中能夠呼叫自定義的動態鏈結庫函式。

第二步,我定義了字串操作的函式(簡單起見,還是採用前面的函式名),返回結果為字串:

libexport_api char *mysum(char *a,char *b)

c#定義匯入定義:

public class ref***m

[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= ref***m. mysum(「12345」, strdest);

執行檢視結果strtmp為「12345」,但是strdest為空。

libexport_api char *mysum(char *a,char *b)

修改c#匯入定義,將串b修改為ref方式:

public class ref***m

[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= ref***m. mysum(「12345」, ref strdest);

執行檢視結果strtmp和strdest均不對,含不可見字元。

再修改c#匯入定義,將charset從auto修改為ansi:

public class ref***m

[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= ref***m. mysum(「12345」, ref strdest);

執行檢視結果strtmp為「12345」,但是串strdest沒有賦值。第二步實現函式返回串,但是在函式出口引數中沒能進行輸出。

再次修改c#匯入定義,將串b修改為引用(ref):

public class ref***m

[dllimport("libencrypt.dll", entrypoint=" mysum ",charset=charset.ansi,callingconvention=callingconvention.stdcall)] public static extern string mysum (string a, ref string b);

執行時呼叫失敗,不能繼續執行。

libexport_api char *mysum(char *a,char **b)

c#匯入定義:

public class ref***m

[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= ref***m. mysum(「12345」, ref strdest);

執行檢視結果strtmp和strdest均為「12345」,呼叫正確。第三步實現了函式出口引數正確輸出結果。

libexport_api int mysum(int a,int b,int *c)

c#匯入的定義:

public class ref***m

[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= ref***m. mysum(2,3, ref c);

執行檢視結果isum 和c均為5,呼叫正確。

經過以上幾個步驟的試驗,基本掌握了如何定義動態庫函式以及如何在c#定義匯入,有此基礎,很快我實現了變長加密函式在c#中的呼叫,至此目標實現。

三、結論

在c#中,呼叫c++編寫動態鏈結庫函式,如果需要出口引數輸出,則需要使用指標,對於字串,則需要使用雙重指標,對於c#的匯入定義,則需要使用引用(ref)定義。

對於函式返回值,c#匯入定義和c++動態庫函式申明定義需要保持一致,否則會出現函式呼叫失敗。

定義匯入時,一定注意charset和callingconvention引數,否則導致呼叫失敗或結果異常。

執行時,動態鏈結庫放在c#程式的目錄下即可,我這裡是乙個c#的動態鏈結庫,兩個動態鏈結庫就在同乙個目錄下執行。

如何在C 中載入自己編寫的動態鏈結庫(DLL)

如何在 c 中載入自己編寫的動態鏈結庫 dll 李偉華 msn liweihua200204 hotmail.com 一 發生的背景 在開發新專案中使用了新的語言開發c 和新的技術方案web service,但是在新專案中,一些舊的模組需要繼續使用,一般是採用c或c 或delphi編寫的,如何利用舊...

如何在C 中載入自己編寫的動態鏈結庫(DLL)

如何在 c 中載入自己編寫的動態鏈結庫 dll 李偉華 msn liweihua200204 hotmail.摘要 本文主要講述如何在 c 中逐步實現載入自己用 c 語言編寫的動態鏈結庫,以及在匯入時如何進行c 和 c 語言的資料型別匹配 c c 動態鏈結庫 載入資料型別匹配 一 發生的背景 在開發...

如何在C 中載入自己編寫的動態鏈結庫(DLL)

一 發生的背景 在開發新專案中使用了新的語言開發c 和新的技術方案web service,但是在新專案中,一些舊的模組需要繼續使用,一般是採用c或c 或delphi編寫的,如何利用舊模組對於開發人員來說,有三種可用方法供選擇 第 一 將c或c 函式用c 徹底改寫一遍,這樣整個專案 比較統一,維護也方...