C++如何调用C#编写的DLL_C++使用CLR进行中介调用教程【跨语言】

2次阅读

会,c++项目直接#include c# dll必然报错,因c# dll是托管程序集(il字节码),非原生dll,编译器不识别;唯一可行路径是用c++/cli作桥梁,通过#using引用、gcnew创建托管对象,并确保clr项目配置、运行时部署均正确。

C++如何调用C#编写的DLL_C++使用CLR进行中介调用教程【跨语言】

C++项目里直接#include C# DLL会报错吗? 会,而且根本过不了编译。C#编译出的DLL是托管程序集(.NET IL字节码),不是windows原生DLL,LoadLibrary加载会失败,#include__declspec(dllimport)完全无效——这不是符号导出问题,是运行时模型不兼容。

真正能走通的路径只有一条:用C++/CLI作为桥梁。它不是“调用DLL”,而是把C#类库当作.NET引用,在混合模式下实例化托管对象。

C++/CLI项目怎么配置才能引用C# DLL? 关键在项目属性和代码层两级设置:

  • 项目类型必须选 CLR Empty Project(不是“Windows 桌面应用”或“静态库”)
  • Configuration Properties → General → Common Language Runtime Support 设为 Common Language Runtime Support (/clr)
  • 右键项目 → Add Reference... → 浏览到你的C# DLL(.NET Framework 4.7.2+ 或 .NET 5+ 都行,但需匹配目标框架)
  • 头文件中用 #using "YourCSharpLib.dll"(注意是#using,不是#include
  • 命名空间要用托管语法:using Namespace YourCSharpNamespace;

漏掉任何一项,都会遇到 Error C3625: 'X': cannot call a native member function on a managed ObjectLNK2028: unresolved Token

从C++代码里怎么安全创建和使用C#类实例? 不能用new原生方式,必须用托管句柄(gcnew)并配合^指针

// 假设C#里有 public class Calculator { public int Add(int a, int b) => a + b; } Calculator^ calc = gcnew Calculator(); int result = calc->Add(3, 4); // 注意是 ->,不是 .

注意事项:

  • 托管对象生命周期由GC管理,不要delete,也不用free
  • 如果C#方法返回String,C++/CLI里拿到的是System::String^,转成原生std::string要用msclr::interop::marshal_as<:string>(s)</:string>(需引用<msclr></msclr>
  • 避免在非CLR项目中直接包含.cpp文件——混合代码只能存在于/clr启用的编译单元里

为什么Release模式下C# DLL总提示“找不到程序集”? 这是部署阶段最常踩的坑:C# DLL本身不带运行时依赖信息,而.NET Framework或.NET Runtime不会自动从C++进程路径加载它。

解决办法只有两个:

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

  • 把C# DLL和它的所有依赖(如Newtonsoft.json.dll)全部复制到C++可执行文件同目录,并确保目标机器装了对应版本的.NET Desktop Runtime(.NET Framework)或 microsoft .NET Runtime(.NET 5+)
  • 或者改用“自包含发布”:在C#项目里设<selfcontained>true</selfcontained>,但这对C++/CLI桥接无意义——因为桥接层仍需运行时支持

最容易被忽略的是:C++/CLI生成的EXE或DLL本身也必须标记为“面向.NET”,它的清单(manifest)里要声明依赖,否则双击运行就弹“未能加载文件或程序集”。

text=ZqhQzanResources