使用P/Invoke调用C++非托管代码需四步:1. 用extern "C"导出C++函数防止名称修饰;2. 在C#中用[DllImport]声明对应方法并指定调用约定;3. 正确映射数据类型如int→Int32、char*→StringBuilder;4. 确保DLL与程序架构匹配并置于可访问路径。

P/Invoke(Platform Invocation Services)是 .NET 提供的一种机制,允许托管代码调用在非托管动态链接库(如 C++ 编写的 DLL)中定义的函数。当你需要使用操作系统 API 或已有 C/C++ 库时,P/Invoke 是一个常用手段。
如何使用 P/Invoke 调用 C++ 非托管代码
要成功调用非托管代码,需完成以下几个步骤:
1. 确保 C++ 函数以 C 方式导出
.NET 通过函数名查找导出函数,而 C++ 存在函数名修饰(name mangling),因此必须使用 extern "C" 来防止修饰,并确保函数按 C 约定导出。
立即学习“C++免费学习笔记(深入)”;
// MathLibrary.h #ifdef MATHLIBRARY_EXPORTS #define MATHLIBRARY_API __declspec(dllexport) #else #define MATHLIBRARY_API __declspec(dllimport) #endifextern "C" MATHLIBRARY_API int Add(int a, int b);
立即学习“C++免费学习笔记(深入)”;
// MathLibrary.cpp #include "MathLibrary.h"extern "C" MATHLIBRARY_API int Add(int a, int b) { return a + b; }
编译后生成 MathLibrary.dll。
2. 在 C# 中声明外部方法
使用 [DllImport] 特性告诉 .NET 这个方法在非托管 DLL 中定义。需指定 DLL 名称和调用约定。
立即学习“C++免费学习笔记(深入)”;
using System; using System.Runtime.InteropServices;class Program { [DllImport("MathLibrary.dll", CallingConvention = CallingConvention.Cdecl)] public static extern int Add(int a, int b);
static void Main() { int result = Add(5, 3); Console.WriteLine("Result: " + result); // 输出: Result: 8 }}
注意: 如果 C++ 导出使用的是 __stdcall,则应改为 CallingConvention.StdCall。上面示例使用 Cdecl,常见于显式导出函数。
3. 处理数据类型映射
托管与非托管类型之间需正确对应。常见映射包括:
- int ↔ Int32
- double ↔ double
- char* ↔ string 或 StringBuilder
- bool ↔ bool(注意调用约定和大小)
例如,导出一个字符串处理函数:
立即学习“C++免费学习笔记(深入)”;
// C++ 代码
extern "C" MATHLIBRARY_API void GetText(char* buffer, int bufferSize) {
strncpy_s(buffer, bufferSize, "Hello from C++!", _TRUNCATE);
}
立即学习“C++免费学习笔记(深入)”;
// C# 调用
[DllImport("MathLibrary.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern void GetText(StringBuilder buffer, int bufferSize);
static void Main() {
StringBuilder sb = new StringBuilder(256);
GetText(sb, sb.Capacity);
Console.WriteLine(sb.ToString()); // 输出: Hello from C++!
}
4. 部署与运行
确保生成的 DLL 与 .NET 程序在同一目录下,或位于系统可找到的路径中(如 PATH)。x64 程序需调用 x64 版本的 DLL,x86 同理,注意平台匹配。
可在项目属性中设置“平台目标”,或使用运行时检测并加载对应架构的 DLL。
基本上就这些。只要导出方式正确、签名匹配、类型映射清晰,P/Invoke 能稳定调用大多数 C/C++ 非托管函数。










