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; 
};
//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;
    };
}
//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);
        }
    }
}

测试结果如下:

这里写图片描述

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

  • 4
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 6
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值