C调用C++的dll方法

C调用C++的dll方法C 调用 C 的 dll 总归可以有两种方法 1 非托管 C 创建的 dll 库 需要用静态方法调用 2 直接使用 CLR 生成托管 C dll 库

C#调用C++的dll总归可以有两种方法:

1、非托管C++创建的dll库,需要用静态方法调用;

2、直接使用CLR,生成托管C++dll库。

托管C++dll库方式

1、打开VS创建C++项目”C++_CScharp_DLL”

![(https://img-blog.csdnimg.cn/8ebcb30014aa46a589cb4d46a97f0dd8.png)

点击确定之后接着点击下一步:

cjxm2

点击完成,C++的项目就新建好了。

2、添加代码文件

右键项目,添加类,如下图所示:

tjdmwj1

添加类之后会打开添加文件对话框,点击添加即可,如下图所示:

在这里插入图片描述

点击确定之后进去下一个对话框,填写文件名Function,如下图所示:

在这里插入图片描述

添加好后会生成h文件和cpp文件,如下图所示:

tjdmwj4

Function类文件代码如下:

//Function.h #pragma once #include  
     public ref class Function { 
    public: Function(void); ~Function(void); int menber; int menberFuncAdd(int a,int b); System::String^ say(System::String^ str); }; //.cpp #include "Function.h" Function::Function(void) { 
    } Function::~Function(void) { 
    } int Function::menberFuncAdd(int a,int b) { 
    return a+b; } System::String^ Function::say(System::String^ str) { 
    return str; } 

填写完后Function.h文件会报错,错误类型如下:

tjdmwj4

这里需要在C++项目里面设置,让动态库受到公共语言运行时的支持。如下图所示:

3、添加测试程序:

using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace Test { 
    class Program { 
    static void Main(string[] args) { 
    Function fun = new Function(); Console.WriteLine(fun.menberFuncAdd(1, 2)); Console.WriteLine(fun.say("Hello World")); Console.ReadKey(); } } } 

非托管C++dll库

非托管模式从功能上来说,只支持函数调用,直接调用C++类库中的公共方法,在被导出的函数前面一定要添加额extern “C来指明导出函数的时候使用C语言方式编译和链接的,这样保证函数定义的名字相同,否则如果默认按C++方式导出,那个函数名字就会变得乱七八糟,我们的程序就无法找到入口点了。

[DllImport(“SampleCppWrapper.dll”)]

private static extern int Add(int n1, int n2);

注意:

在编译C++DLL之前,需要做以下配置,在项目属性对话框中选择”C/C++“|“Advanced”,将Compile AS 选项的值改为”C++”。然后确定,并编译。

在这里插入图片描述

以下为摘录例子:

1、 调用C++类库中的类的方法

C#不能直接调用C++类库中的类,需要一种变通的解决方式,通过再做一个C++类库把要调用的类成员方法暴露出来,比如下面这个C++类:

//SampleCppClass.h #pragma once class __declspec(dllexport) SampleCppClass { 
    public: SampleCppClass(void); ~SampleCppClass(void); int Add(int n1, int n2); int Sub(int n1, int n2); }; //SampleCppClass.cpp #include "SampleCppClass.h" SampleCppClass::SampleCppClass(void) { 
    } SampleCppClass::~SampleCppClass(void) { 
    } int SampleCppClass::Add(int n1, int n2) { 
    return n1 + n2; } int SampleCppClass::Sub(int n1, int n2) { 
    return n1 - n2; } 

我们要调用SampleCppClass中的Add和Sub两个方法,所以我们再写一个C++类库,通过公共方法间接调用类成员方法:

//SampleCppWrapper.h #pragma once #include "..\SampleCppClass\SampleCppClass.h" namespace SampleCppWrapper { 
    extern "C" __declspec(dllexport) int __stdcall Add(int n1, int n2); extern "C" __declspec(dllexport) int __stdcall Sub(int n1, int n2); } //SampleCppWrapper.cpp #include "SampleCppWrapper.h" namespace SampleCppWrapper { 
    SampleCppClass* g_pObj = new SampleCppClass(); int __stdcall Add(int n1, int n2) { 
    return g_pObj->Add(n1, n2); } int __stdcall Sub(int n1, int n2) { 
    return g_pObj->Sub(n1, n2); } } 

在C#中,再调用SampleCppWrapper.dll中的公共方法:

[DllImport("SampleCppWrapper.dll")] private static extern int Add(int n1, int n2); [DllImport("SampleCppWrapper.dll")] private static extern int Sub(int n1, int n2); 

3、 使用C++类库中的回调函数

C++的回调函数是一种事件响应机制,和C#的委托相似,比如一个C++类中的回调函数:

// SampleCppClass.h #pragma once typedef void (*LoopCallback)(void* pContext); class __declspec(dllexport) SampleCppClass { 
    public: SampleCppClass(void); ~SampleCppClass(void); void SetCallbackFunc(LoopCallback callback); void SetCallbackContext(void* pContext); void Loop(); private: LoopCallback m_callback; void* m_pContext; }; // SampleCppClass.cpp #include "SampleCppClass.h" SampleCppClass::SampleCppClass(void) { 
    } SampleCppClass::~SampleCppClass(void) { 
    } void SampleCppClass::SetCallbackFunc(LoopCallback callback) { 
    m_callback = callback; } void SampleCppClass::SetCallbackContext(void* pContext) { 
    m_pContext = pContext; } void SampleCppClass::Loop() { 
    for (int i=0; i<10; i++) { 
    if (m_callback != NULL) { 
    m_callback(m_pContext); } } } 

导出方法文件中添加:

//.h #pragma once #include "..\SampleCppClass\SampleCppClass.h" namespace SampleCppWrapper { 
    typedef void (__stdcall *LoopCallbackWrapper)(void* pContext); extern "C" __declspec(dllexport) void __stdcall SetCallbackFunc(LoopCallbackWrapper callback); extern "C" __declspec(dllexport) void __stdcall SetCallbackContext(void* pContext); extern "C" __declspec(dllexport) void __stdcall Loop(); } // .CPP #include "SampleCppWrapper.h" namespace SampleCppWrapper { 
    LoopCallbackWrapper g_callbackWrapper; SampleCppClass* g_pObj = new SampleCppClass(); void LoopCallbackFunc(void* pContext); void __stdcall SetCallbackFunc(LoopCallbackWrapper callback) { 
    g_callbackWrapper = callback; g_pObj->SetCallbackFunc(LoopCallbackFunc); } void __stdcall SetCallbackContext(void* pContext) { 
    g_pObj->SetCallbackContext(pContext); } void __stdcall Loop() { 
    g_pObj->Loop(); } void LoopCallbackFunc(void* pContext) { 
    if (g_callbackWrapper != NULL) { 
    g_callbackWrapper(pContext); } } } 

C#中调用:

using System; using System.Runtime.InteropServices; using System.Windows.Forms; namespace SampleCsTest { 
    public partial class Form1 : Form { 
    [StructLayout(LayoutKind.Sequential)] private class Context { 
    public Form1 Form { 
    get; set; } } private delegate void LoopCallbackHandler(IntPtr pContext); private static LoopCallbackHandler callback = LoopCallback; [DllImport("SampleCppWrapper.dll")] private static extern void SetCallbackFunc(LoopCallbackHandler callback); [DllImport("SampleCppWrapper.dll")] private static extern void SetCallbackContext(IntPtr pContext); [DllImport("SampleCppWrapper.dll")] private static extern void Loop(); private Context ctx = new Context(); public Form1() { 
    InitializeComponent(); } private void Form1_Load(object sender, EventArgs e) { 
    SetCallbackFunc(callback); ctx.Form = this; IntPtr ptr = Marshal.AllocHGlobal(Marshal.SizeOf(ctx)); Marshal.StructureToPtr(ctx, ptr, false); SetCallbackContext(ptr); } private void button1_Click(object sender, EventArgs e) { 
    Loop(); } private static void LoopCallback(IntPtr pContext) { 
    Context ctx = (Context)Marshal.PtrToStructure(pContext, typeof(Context)); ctx.Form.textBox1.Text += "callback" + Environment.NewLine; } } } 

以上为非托管方式简单参数的传递,实际使用过程中,可能参数类型会复杂很多,这牵涉到C# C++之间的参数转换及C#语法对托管代码的编写,具体做一些项目时,肯定会比例子情况复杂的多,那就需要对各种参数传递及转换好好了解一番,如果解决了各种情况参数传递问题,基本C#调用非托管C++dll没有其它复杂问题。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/206927.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月19日 下午3:11
下一篇 2026年3月19日 下午3:11


相关推荐

  • 手背静脉识别的图像处理算法

    手背静脉识别的图像处理算法手背静脉识别的图像处理算法题目内容及要求手背静脉识别技术作为一种全新的特征识别技术,相比于传统的生物识别技术(如指纹识别)具有许多明显的优势,然而对于该技术的研究尚处于刚刚起步阶段,使用计算机来直接进行静脉识别与身份匹配仍然较为困难,为了方便后续特征识别,提高静脉识别的准确度和优越性,有必要对获取的静脉图像进行一系列处理,得到静脉的骨架结构。题目主要要求为:1.对采集图像进行背景去除,取得…

    2022年5月16日
    49
  • SWUFE Agent智能体

    SWUFE Agent智能体

    2026年3月16日
    1
  • Mac安装双系统「建议收藏」

    Mac安装双系统「建议收藏」Mac安装双系统

    2022年10月5日
    6
  • JDBC batch批处理Statement executeBatch 具体解释

    JDBC batch批处理Statement executeBatch 具体解释

    2021年11月30日
    44
  • 初次尝试使用VisualSFM记录

    初次尝试使用VisualSFM记录  对于SFM一直觉得高大上又神秘,一年前粗略的了解过一下,今年有时间尝试深入了解SFM,对于初学者来说,VisualSFM真的时非常好的感受SFM的免费软件,于是通过其他博客的指导(没去看官方的英文版,容易犯困),做了一个3D模型出来,感觉还挺好玩,记录一下。  实际上,VisualSFM只做了一部分工作,优化是用MeshLab做的,所以分两个环节:   1.VisualSF…

    2022年6月20日
    26
  • 【Python基础】PyCharm配置Python虚拟环境详解[通俗易懂]

    【Python基础】PyCharm配置Python虚拟环境详解[通俗易懂]目录一、基础介绍1.1基础介绍1.2配置现状二、步骤详解2.1新建项目2.2查看虚拟环境2.3安装需要的包2.4验证安装三、一、基础介绍1.1基础介绍Python的版本众多,而且其内部的库Package也五花八门,这就导致在同时进行几个项目时,对库的依赖存在很大的问题。这个时候就牵涉到对Python以及依赖库的版本管理,方便进行开发,就需要进行虚拟环境的配置。一方面:我们初学python的时候,下载第三方库的时候其实是在全局或者是整个系统中都可以使用,但对于一些项目来说,需要的库可能是

    2022年8月29日
    7

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号