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

使用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 约定导出。

// MathLibrary.h
#ifdef MATHLIBRARY_EXPORTS
#define MATHLIBRARY_API __declspec(dllexport)
#else
#define MATHLIBRARY_API __declspec(dllimport)
#endif

extern "C" MATHLIBRARY_API int Add(int a, int b);

// 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 名称和调用约定。

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. 处理数据类型映射

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

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

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

// C++ 代码
extern "C" MATHLIBRARY_API void GetText(char* buffer, int bufferSize) {
    strncpy_s(buffer, bufferSize, "Hello from C++!", _TRUNCATE);
}

// 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++ 非托管函数。