答案是P/Invoke适用于简单C风格函数调用,C++/CLI适合复杂C++类交互;前者轻量但受限,后者灵活但需托管包装。
在C#中调用C++编写的DLL,主要有两种方式:P/Invoke(平台调用)和C++/CLI。选择哪种方式取决于你的需求、性能要求以及是否需要跨语言复杂类型交互。
当C++ DLL以C接口形式导出函数(即使用extern "C"防止C++名称修饰),并且传递的是基本数据类型或简单结构体时,推荐使用P/Invoke。
步骤如下:
C++ DLL 示例(mycppdll.h):
extern "C" {C# 调用代码:
using System;class Program {
[DllImport("mycppdll.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern int Add(int a, int b);
[DllImport("mycppdll.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern void GetString(StringBuilder buffer, int bufferSize);
static void Main() {
int result = Add(3, 5);
Console.WriteLine("Add result: " + result);
var sb = new StringBuilder(256);
GetString(sb, sb.Capacity);
Console.WriteLine("String from C++: " + sb.ToString());
}
}
注意事项:
当你需要调用C++类、模板、STL容器,或者涉及复杂的对象生命周期管理时,P/Invoke难以胜任。此时应使用C++/CLI创建一个“托管包装器”DLL。
优点:
实现步骤:
C++/CLI 包装示例(Wrapper.h):
// 原生C++类
ly(int x, int y);// 托管包装类
public ref class ManagedMath {
private:
NativeMath* native;
public:
ManagedMath() {
native = new NativeMath();
}
~ManagedMath() {
delete native;
}
int Multiply(int x, int y) {
return native->Multiply(x, y);
}
};
C# 调用代码:
class Program {关键点:
根据实际场景判断:
基本上就这些。两种方式都能有效打通C#与C++的壁垒,关键是理解各自边界和适用范围。