当前位置: 首页 > java >正文

C++项目中调用C#DLL的的方式

C++项目中调用C#DLL的的方式

  • 方法一:使用COM技术
  • 方法二:使用C++/CLI
  • 方法三:使用P/Invoke(适用于C#导出非托管接口)

在C++中调用C#编写的DLL,通常需要借助COM(Component Object Model)技术或者通过C++/CLI(C++托管扩展)来实现。以下是两种常见的方法:

方法一:使用COM技术

1.在C#中创建COM可见的类库
在C#项目中,确保类库项目属性中的“Make assembly COM-Visible”选项被勾选。这会为类库生成一个GUID,并将其注册为COM组件。
例如,创建一个C#类库项目MyCSharpLibrary,并添加一个类MyClass:
下面展示一些 内联代码片

using System;
using System.Runtime.InteropServices;[ComVisible(true)]
[Guid("YOUR-GUID-HERE")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IMyClass
{void MyMethod();
}[ComVisible(true)]
[Guid("YOUR-CLASS-GUID-HERE")]
public class MyClass : IMyClass
{public void MyMethod(){Console.WriteLine("Hello from C#!");}
}

生成DLL后,使用regasm工具将其注册为COM组件:
下面展示一些 内联代码片

下面展示一些 内联代码片

regasm MyCSharpLibrary.dll /codebase

2.在C++中调用COM组件
在C++代码中,通过COM接口调用C#类库中的方法:
下面展示一些 内联代码片

#include <iostream>
#import "MyCSharpLibrary.tlb" // 导入类型库文件int main()
{CoInitialize(NULL); // 初始化COM库IMyClassPtr myClass(__uuidof(MyClass)); // 创建COM对象myClass->MyMethod(); // 调用方法CoUninitialize(); // 释放COM库return 0;
}

注意:需要确保C++项目中链接了ole32.lib和oleaut32.lib。

方法二:使用C++/CLI

C++/CLI是一种混合编程语言,允许在C++代码中直接使用托管代码(如C#代码)。
创建C++/CLI项目
在Visual Studio中创建一个C++/CLI项目,例如MyCppCliWrapper。
在C++/CLI中引用C# DLL
在C++/CLI项目中添加对C# DLL的引用。
创建一个托管类来封装C#类的功能:
下面展示一些 内联代码片

// MyCppCliWrapper.h
#pragma onceusing namespace System;public ref class MyCppCliWrapper
{
public:void CallCSharpMethod();
};
// MyCppCliWrapper.cpp
#include "MyCppCliWrapper.h"
#include "MyCSharpLibrary.h" // 引用C#类库void MyCppCliWrapper::CallCSharpMethod()
{MyCSharpLibrary::MyClass^ myClass = gcnew MyCSharpLibrary::MyClass();myClass->MyMethod();
}

3.在C++代码中调用C++/CLI封装
在C++代码中调用C++/CLI封装的函数:
下面展示一些 内联代码片

#include <iostream>
#include "MyCppCliWrapper.h"int main()
{MyCppCliWrapper^ wrapper = gcnew MyCppCliWrapper();wrapper->CallCSharpMethod();return 0;
}

方法三:使用P/Invoke(适用于C#导出非托管接口)

如果C# DLL中导出了非托管接口(例如通过DllImport或[UnmanagedCallersOnly]),可以直接在C++中通过函数指针调用。
在C#中导出非托管接口
使用[UnmanagedCallersOnly]属性导出方法:
下面展示一些 内联代码片

using System;
using System.Runtime.InteropServices;public class MyClass
{[UnmanagedCallersOnly(EntryPoint = "MyMethod")]public static void MyMethod(){Console.WriteLine("Hello from C#!");}
}

在C++中调用导出的函数
加载DLL并获取函数指针:
下面展示一些 内联代码片

#include <iostream>
#include <windows.h>typedef void (*MyMethodFunc)();int main()
{HMODULE hModule = LoadLibrary("MyCSharpLibrary.dll");if (hModule){MyMethodFunc myMethod = (MyMethodFunc)GetProcAddress(hModule, "MyMethod");if (myMethod){myMethod();}FreeLibrary(hModule);}return 0;
}

总结
COM技术:适合需要跨语言调用的场景,但需要额外的注册和配置。
C++/CLI:适合需要在C++中直接调用托管代码的场景,代码更简洁。
P/Invoke:适合C#导出非托管接口的场景,调用方式更接近原生C++。

http://www.xdnf.cn/news/9003.html

相关文章:

  • 人工智能工程师学习路线总结(下)
  • pytorch 15.2 学习率调度在PyTorch中的实现方法
  • 0526漏洞原理:漏洞验证-信息收集笔记(BurpSuite Professional,fofa,BUUCTF)
  • 互联网大厂Java求职面试:AI大模型与云原生架构融合中的挑战
  • ai学习--python部分-1.变量名及命名空间的存储
  • django三级联动
  • MongoDB 快速整合 SpringBoot 示例
  • 系统架构中的限流实践:构建多层防护体系(二)
  • 文件管理(第八章、九)
  • Linux常见设备
  • EPT(Efficient Prompt Tuning)方法,旨在解决提示调优(Prompt Tuning)中效率与准确性平衡和跨任务一致性的问题
  • 基于大模型的急性腐蚀性胃炎风险预测与诊疗方案研究报告
  • 【保姆级教程】TortoiseGit安装和Visual Studio2019插件配置详细说明
  • leetcode98.验证二叉搜索树:迭代法中序遍历与栈操作的深度剖析
  • noc多核芯片设计:booksim仿真从入门到精通2Router 类型及路由算法修改
  • 28. 自动化测试开发框架拓展之邮件模块开发
  • Google 发布AI 编程工具Jules
  • Kaggle-Predict Calorie Expenditure-(回归+xgb+cat+lgb+模型融合)
  • 成都鼎讯便携式雷达信号模拟器:重塑电磁训练新生态
  • 【鸿蒙开发】Hi3861学习笔记-雨滴传感器
  • node入门:安装和npm使用
  • C语言利用Windows Portable Devices API访问安卓设备文件
  • 什么是HTTP HTTP 和 HTTPS 的区别
  • 视频画质等级
  • openpi π₀ 项目部署运行逻辑(三)——策略推理服务器 serve_policy.py
  • 中小企业AI算力如何选?【显卡租赁】VS【自建服务器】
  • 语音识别——文本转语音
  • 5.26 day 29
  • 论文阅读:Self-Planning Code Generation with Large Language Models
  • AOSP编译错误