C#调用C++编写的DLL,主要靠P/Invoke(Platform Invocation Services),核心是用
[DllImport]声明外部函数,并确保C++导出函数符合C调用约定、数据类型兼容、内存管理明确。
确保C++ DLL正确导出函数
C++代码必须显式导出函数,推荐使用
extern "C"防止C++名字修饰(name mangling),并指定
__declspec(dllexport): 在头文件中声明:extern "C" __declspec(dllexport) int Add(int a, int b); 实现时保持C链接:extern "C" __declspec(dllexport) int Add(int a, int b) { return a + b; } 避免类、STL容器、异常跨DLL边界;只导出简单函数或C风格接口
C#中用DllImport声明并调用
在C#中用
[DllImport]标记静态方法,指定DLL文件名、调用约定和字符编码: [DllImport("MyNative.dll", CallingConvention = CallingConvention.Cdecl)] 若函数参数含字符串,注意
CharSet(如
CharSet = CharSet.Ansi或
CharSet.Unicode) 简单调用示例:int result = Add(3, 5);
处理复杂数据类型和内存管理
结构体、数组、字符串、指针需特别注意布局与生命周期:
立即学习“C++免费学习笔记(深入)”;
结构体加[StructLayout(LayoutKind.Sequential)],字段顺序和对齐要与C++一致 字符串传入建议用
string+
[MarshalAs(UnmanagedType.LPStr)];传出缓冲区用
StringBuilder避免C++分配、C#释放(或反之);如需跨DLL内存操作,统一用
Marshal.AllocHGlobal或导出配套的释放函数
调试常见问题
调用失败常因路径、位数不匹配或签名不一致:
DLL必须和C#程序位数一致(x86/x64/AnyCPU需匹配) DLL放在exe同目录、系统路径或PATH中;也可用绝对路径测试 用
Dependency Walker或
dumpbin /exports MyNative.dll确认函数名是否导出成功 异常提示“找不到入口点”多半是名字修饰或调用约定不对;“尝试读取或写入受保护的内存”多因指针/内存越界
基本上就这些。P/Invoke不复杂但容易忽略细节,关键是两边接口对齐、约定统一、内存可控。
