C# 中静态调用C++dll 和C# 中动态调用C++dll
在最近的項目中,牽涉到項目源代碼保密問題,由于代碼是C#寫的,容易被反編譯,因此決定抽取核心算法部分使用C++編寫,C++到目前為止好像還不能被很好的反編譯,當然如果你是反匯編高手的話,也許還是有可能反編譯。這樣一來,就涉及C#托管代碼與C++非托管代碼互相調(diào)用,于是調(diào)查了一些資料,順便與大家分享一下:
一. C# 中靜態(tài)調(diào)用C++動態(tài)鏈接
?
??? 1. 建立VC工程CppDemo,建立的時候選擇Win32 Console(dll),選擇Dll。
??? 2. 在DllDemo.cpp文件中添加這些代碼。
Codeextern?"C"?__declspec(dllexport)?int?Add(int?a,int?b)
{
????
?????return?a+b;
}
??? 3. 編譯工程。
??? 4. 建立新的C#工程,選擇Console應用程序,建立測試程序InteropDemo
??? 5. 在Program.cs中添加引用:using System.Runtime.InteropServices;
??? 6. 在pulic class Program添加如下代碼:?
?
Codeusing?System;
using?System.Collections.Generic;
using?System.Text;
using?System.Runtime.InteropServices;
namespace?InteropDemo
{
????class?Program
????{
????????[DllImport("CppDemo.dll",?EntryPoint?=?"Add",?ExactSpelling?=?false,?CallingConvention?=?CallingConvention.Cdecl)]
????????public?static?extern?int?Add(int?a,?int?b);?//DllImport請參照MSDN
????????static?void?Main(string[]?args)
????????{
????????????Console.WriteLine(Add(1,?2));
????????????Console.Read();
????????}
????}
}
?? 好了,現(xiàn)在您可以測試Add程序了,是不是可以在C# 中調(diào)用C++動態(tài)鏈接了,當然這是靜態(tài)調(diào)用,需要將CppDemo編譯生成的Dll放在DllDemo程序的Bin目錄下
二. C# 中動態(tài)調(diào)用C++動態(tài)鏈接
?在第一節(jié)中,講了靜態(tài)調(diào)用C++動態(tài)鏈接,由于Dll路徑的限制,使用的不是很方便,C#中我們經(jīng)常通過配置動態(tài)的調(diào)用托管Dll,例如常用的一些設計模式:Abstract Factory, Provider, Strategy模式等等,那么是不是也可以這樣動態(tài)調(diào)用C++動態(tài)鏈接呢?只要您還記得在C++中,通過LoadLibrary, GetProcess, FreeLibrary這幾個函數(shù)是可以動態(tài)調(diào)用動態(tài)鏈接的(它們包含在kernel32.dll中),那么問題迎刃而解了,下面我們一步一步實驗
??? 1.? 將kernel32中的幾個方法封裝成本地調(diào)用類NativeMethod
Codeusing?System;
using?System.Collections.Generic;
using?System.Text;
using?System.Runtime.InteropServices;
namespace?InteropDemo
{
????public?static?class?NativeMethod
????{
????????[DllImport("kernel32.dll",?EntryPoint?=?"LoadLibrary")]
????????public?static?extern?int?LoadLibrary(
????????????[MarshalAs(UnmanagedType.LPStr)]?string?lpLibFileName);
????????[DllImport("kernel32.dll",?EntryPoint?=?"GetProcAddress")]
????????public?static?extern?IntPtr?GetProcAddress(int?hModule,
????????????[MarshalAs(UnmanagedType.LPStr)]?string?lpProcName);
????????[DllImport("kernel32.dll",?EntryPoint?=?"FreeLibrary")]
????????public?static?extern?bool?FreeLibrary(int?hModule);
????}
}
?
??? 2. 使用NativeMethod類動態(tài)讀取C++Dll,獲得函數(shù)指針,并且將指針封裝成C#中的委托。原因很簡單,C#中已經(jīng)不能使用指針了,如下 ????????
??????????? int hModule = NativeMethod.LoadLibrary(@"c:"CppDemo.dll");
??????????? IntPtr intPtr = NativeMethod.GetProcAddress(hModule, "Add");
詳細請參見代碼
Codeusing?System;
using?System.Collections.Generic;
using?System.Text;
using?System.Runtime.InteropServices;
namespace?InteropDemo
{
????class?Program
????{
????????//[DllImport("CppDemo.dll",?EntryPoint?=?"Add",?ExactSpelling?=?false,?CallingConvention?=?CallingConvention.Cdecl)]
????????//public?static?extern?int?Add(int?a,?int?b);?//DllImport請參照MSDN
????????static?void?Main(string[]?args)
????????{
????????????//1.?動態(tài)加載C++?Dll
????????????int?hModule?=?NativeMethod.LoadLibrary(@"c:\CppDemo.dll");
????????????if?(hModule?==?0)?return;
????????????//2.?讀取函數(shù)指針
????????????IntPtr?intPtr?=?NativeMethod.GetProcAddress(hModule,?"Add");
????????????//3.?將函數(shù)指針封裝成委托
????????????Add?addFunction?=?(Add)Marshal.GetDelegateForFunctionPointer(intPtr,?typeof(Add));
????????????//4.?測試
????????????Console.WriteLine(addFunction(1,?2));
????????????Console.Read();
????????}
????????///?<summary>
????????///?函數(shù)指針
????????///?</summary>
????????///?<param?name="a"></param>
????????///?<param?name="b"></param>
????????///?<returns></returns>
????????delegate?int?Add(int?a,?int?b);
????}
}
?
?
?
通過如上兩個例子,我們可以在C#中動態(tài)或者靜態(tài)的調(diào)用C++寫的代碼了.
轉(zhuǎn)自:http://www.cnblogs.com/Jianchidaodi/archive/2009/03/09/1407270.html
?
轉(zhuǎn)載于:https://www.cnblogs.com/newstart/archive/2012/10/24/2736780.html
總結
以上是生活随笔為你收集整理的C# 中静态调用C++dll 和C# 中动态调用C++dll的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: JAVA_JDK下载与安装教程
- 下一篇: SBuild 0.1.4 发布,基于 S