C++调用C#注册的回调函数

本文主要讲解C#注册回调函数提供给C++调用,用于异步通知机制。这在编程过程中是经常需要用到的模式。
此过程需要创建三个工程:

  1. C++动态库工程
  2. C++ CLR 托管动态库工程
  3. C# 测试工程
    接着前文C++封装成托管模式供C#调用过程创建的C++ CLR托管工程,在此基础上再创建一个C++动态库工程,此工程创建过程比较简单在此不再详细说明,只注意一点如图。
    在这里插入图片描述

一、C++动态库工程

工程结构图:

//CPPClass.h
#ifdef CPPCLASS_EXPORTS
#define CPPCLASS_API __declspec(dllexport)
#else
#define CPPCLASS_API __declspec(dllimport)
#endif

typedef void (_stdcall *CB_FUNCTION_CALLBACK)(int IntputDataClr);

// 此类是从 CPPClass.dll 导出的
class CPPCLASS_API CCPPClass {
public:
CCPPClass(void);
bool RegisterFunctionCallback(CB_FUNCTION_CALLBACK cbFunctionCallback);
bool InvokeCallbackFunction();
private:
CB_FUNCTION_CALLBACK mCallback;
};123456789101112131415161718

//CPPClass.cpp
#include “stdafx.h”
#include “CPPClass.h”

CCPPClass::CCPPClass()
:mCallback(nullptr)
{
return;
}

bool CCPPClass::RegisterFunctionCallback(CB_FUNCTION_CALLBACK cbFunctionCallback)
{
if(nullptr != cbFunctionCallback){
mCallback = cbFunctionCallback;
return true;
}else{
mCallback = nullptr;
return false;
}
}

bool CCPPClass::InvokeCallbackFunction()
{
if(nullptr != mCallback){
mCallback(1024); //调用回调函数,把1024传给C#
return true;
}else{
return false;
}
}

二、 C++ CLR 托管动态库工程
此工程的创建在前面文章已经详细说明,此处不在叙述。
工程结构图:
在这里插入图片描述

//CppCLR.h
#include “CPPClass.h”
#include <msclr\marshal_cppstd.h>
#include <msclr/marshal.h>
#include <msclr/marshal_windows.h>
#include <msclr/marshal_cppstd.h>
#include <msclr/marshal.h>
using namespace System;
using namespace msclr::interop;
using namespace System::Runtime::InteropServices;

namespace CppCLRNameSpace {
public delegate void FunctionCallback(int IntputDataClr);
public ref class ClassCLR
{
public:
ClassCLR();
~ClassCLR();
bool RegisterFunctionCallback(FunctionCallback^ cbFunctionCallback);
bool InvokeCallbackFunction();

private:
    CCPPClass *mCPPClass;
};

}12345678910111213141516171819202122232425

//CppCLR.cpp
#include “stdafx.h”
#include “CppCLR.h”
using namespace CppCLRNameSpace;
ClassCLR::ClassCLR()
{
mCPPClass = nullptr;
mCPPClass = new CCPPClass();
}

ClassCLR::~ClassCLR()
{
delete mCPPClass;
mCPPClass = nullptr;
}

bool ClassCLR::RegisterFunctionCallback(FunctionCallback^ cbFunctionCallback)
{
IntPtr c_cbDownloadCallback = Marshal::GetFunctionPointerForDelegate(cbFunctionCallback);//可将pvFun强制转化为void*,再强制转化为FUN类型
return mCPPClass->RegisterFunctionCallback(reinterpret_cast<CB_FUNCTION_CALLBACK>(c_cbDownloadCallback.ToInt32()));
}

bool ClassCLR::InvokeCallbackFunction()
{
return mCPPClass->InvokeCallbackFunction();
}

三、C# 测试工程
创建一个C#的控制台应用程序用于测试,此工程中需要引用CppCLR.dll托管动态库,还需要添加using CppCLRNameSpace;才能够使用ClassCLR托管类。
注意:还需要把C++生成的CPPClass.dll添加到C#的bin执行目录下,因为此库需要被CppCLR.dll托管库引用
工程结构图:
在这里插入图片描述

//Program.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using CppCLRNameSpace;
namespace TestCppCLR
{
class Program
{
static void Main(string[] args)
{
ClassCLR mClassCLR = new ClassCLR();
FunctionCallback mFunctionCallback;
ClassTemp mClassTemp = new ClassTemp();
mFunctionCallback = new FunctionCallback(mClassTemp.Functions);
mClassCLR.RegisterFunctionCallback(mFunctionCallback);
mClassCLR.InvokeCallbackFunction();
Thread.Sleep(100000);
}
}

public class ClassTemp
{
    public void Functions(int InputData)
    {
        Console.WriteLine(InputData);
    }
}

}1234567891011121314151617181920212223242526272829303132

测试结果如下:
在这里插入图片描述

总结:
对于一些比较复杂类型的数据结构则需要考虑到C#与C++数据类型之间的差异,例如string等,因为C#与C++的string不能够相对应,不能直接传递,需要变换成char*。关于更多数据类型转换可以参考另外一篇博文。C++托管与C#中的数据类型转换

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值