.NET中的P/Invoke是什么?如何调用C++编写的非托管代码?

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

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)
#endif
<p>extern "C" MATHLIBRARY_API int Add(int a, int b);</p>

立即学习“C++免费学习笔记(深入)”;

// MathLibrary.cpp
#include "MathLibrary.h"
<p>extern "C" MATHLIBRARY_API int Add(int a, int b) {
return a + b;
}</p>

编译后生成 MathLibrary.dll。

2. 在 C# 中声明外部方法

使用 [DllImport] 特性告诉 .NET 这个方法在非托管 DLL 中定义。需指定 DLL 名称和调用约定。

立即学习“C++免费学习笔记(深入)”;

using System;
using System.Runtime.InteropServices;
<p>class Program {
[DllImport("MathLibrary.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern int Add(int a, int b);</p><pre class='brush:php;toolbar:false;'>static void Main() {
    int result = Add(5, 3);
    Console.WriteLine("Result: " + result); // 输出: Result: 8
}

}

注意: 如果 C++ 导出使用的是 __stdcall,则应改为 CallingConvention.StdCall。上面示例使用 Cdecl,常见于显式导出函数。

3. 处理数据类型映射

托管与非托管类型之间需正确对应。常见映射包括:

intInt32 doubledouble char*stringStringBuilder boolbool(注意调用约定和大小)

例如,导出一个字符串处理函数:

立即学习“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);
<p>static void Main() {
StringBuilder sb = new StringBuilder(256);
GetText(sb, sb.Capacity);
Console.WriteLine(sb.ToString()); // 输出: Hello from C++!
}</p>

4. 部署与运行

确保生成的 DLL 与 .NET 程序在同一目录下,或位于系统可找到的路径中(如 PATH)。x64 程序需调用 x64 版本的 DLL,x86 同理,注意平台匹配。

可在项目属性中设置“平台目标”,或使用运行时检测并加载对应架构的 DLL。

基本上就这些。只要导出方式正确、签名匹配、类型映射清晰,P/Invoke 能稳定调用大多数 C/C++ 非托管函数。

相关推荐