C#如何调用C++写的DLL P/Invoke平台调用方法

来源:这里教程网 时间:2026-02-21 17:34:34 作者:

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不复杂但容易忽略细节,关键是两边接口对齐、约定统一、内存可控。

相关推荐