C#如何调用C++的DLL C# P/Invoke平台调用非托管代码

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

在C#中调用C++编写的DLL,通常使用P/Invoke(Platform Invoke)技术。它允许托管代码调用非托管DLL中的函数,比如用C或C++编写的Win32 API或自定义DLL。下面详细介绍如何实现这一过程。

1. 编写C++ DLL导出函数

首先确保你的C++ DLL正确导出了可供外部调用的函数。推荐使用 extern "C" 防止C++命名修饰,并使用 __declspec(dllexport) 导出函数。

示例C++代码(MyCppDll.cpp):

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

extern "C" __declspec(dllexport) int Add(int a, int b) {
    return a + b;
}
<p>extern "C" __declspec(dllexport) void PrintMessage(const char* msg) {
printf("Message: %s\n", msg);
}
</p>

编译为DLL后,例如生成 MyCppDll.dll

2. 在C#中声明DLL导入函数

使用 [DllImport] 特性声明要调用的函数。注意指定DLL名称和参数类型映射。

C#代码示例:

using System;
using System.Runtime.InteropServices;
<p>class Program
{
// 声明Add函数
[DllImport("MyCppDll.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern int Add(int a, int b);</p><pre class="brush:php;toolbar:false;">// 声明PrintMessage函数
[DllImport("MyCppDll.dll", CallingConvention = CallingConvention.Cdecl, 
           CharSet = CharSet.Ansi)]
public static extern void PrintMessage(string msg);
static void Main()
{
    int result = Add(5, 3);
    Console.WriteLine("5 + 3 = " + result);
    PrintMessage("Hello from C#!");
}

}

关键点说明:

DllImport属性:指定DLL文件名,无需路径则要求DLL在运行目录或系统路径中。 CallingConvention:C++默认使用Cdecl调用约定,需显式指定。 CharSet:字符串编码方式,Ansi对应char*,Unicode对应wchar_t*。

3. 处理复杂数据类型

当需要传递结构体或数组时,必须在C#中定义对应的布局。

例如C++结构体:

struct Point {
    int x;
    int y;
};
extern "C" __declspec(dllexport) double Distance(Point p1, Point p2);

C#中对应声明:

[StructLayout(LayoutKind.Sequential)]
public struct Point
{
    public int x;
    public int y;
}
<p>[DllImport("MyCppDll.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern double Distance(Point p1, Point p2);
</p>

如果传指针,可使用 refout 参数,或配合 Marshal 类手动管理内存。

4. 调试与常见问题

找不到DLL:确认DLL位于输出目录(如bin\Debug),或使用绝对路径。 EntryPointNotFoundException:检查函数名是否被C++修饰,使用extern "C"避免名字损坏。 平台匹配:确保C#项目目标平台(x86/x64)与DLL一致。 字符集错误:若C++接收wchar_t*,应设CharSet.Unicode并使用IntPtr或MarshalAs处理。

可使用工具如dumpbin /exports MyCppDll.dll查看实际导出函数名。

基本上就这些。只要DLL导出规范,C#通过P/Invoke调用并不复杂,但要注意类型映射和调用约定的一致性。

相关推荐