C# Winform 使用委托实现C++中回调函数的功能

C# Winform 使用委托实现C++中回调函数的功能

在项目中遇到了使用C#调用C++封装的接口,其中C++接口有一个回调函数的参数。参考对比后,在C#中是使用委托(delegate)来实现类似的功能。

下面使用一个示例来介绍具体的使用方式:

第一步:新建C++动态连接库

1.首先新建一个C++动态库程序,如下图所示:

在这里插入图片描述

并且在dllmain.cpp文件中添加如下导数函数代码:

typedef int (*FUNC)(int a, int b);
extern "C"  __declspec(dllexport) void _stdcall AddNum(int a,int b,FUNC callback)
{
    callback(a, b);
}

该函数传入两个整数,并且调用外部的回调函数对这两个整数运算处理。

运行程序生成了相应的DLL

第二步:新建WinForm项目

我新建了一个简单的WinForm程序的界面如下:

微信截图_20250209232909

由用户输入两个数值,选择相应的运算符,分别有+,-,*,/ 四种运算符,点击计算按钮就得到运算结果。上面我填了10 * 20 = 200;

具体代码如下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Reflection.Emit;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using static System.Net.Mime.MediaTypeNames;

namespace WinFormCallBackTest
{
    public partial class Form1 : Form
    {
        public delegate int CallBackFunc(int a, int b);
        CallBackFunc callback = null;

        [DllImport("CallBackDLLTest.dll", CallingConvention = CallingConvention.StdCall)]
        public static extern void AddNum(int a,int b,CallBackFunc callback);

        public Form1()
        {
            InitializeComponent();
            // 设置窗体启动位置为屏幕中央
            this.StartPosition = FormStartPosition.CenterScreen;

            callback = new CallBackFunc(CallBackAddNum);

        }

        public int CallBackAddNum(int a, int b)
        {
            int nSum = 0;
            if (comboBox1.Text == "+")
            {
                nSum = a + b;
            }
            else if (comboBox1.Text == "-")
            {
                nSum = a - b;
            }
            else if (comboBox1.Text == "*")
            {
                nSum = a * b;
            }
            else if (comboBox1.Text == "/")
            {
                nSum = a / b;
            }

            // 检查是否需要跨线程操作
            if (textBox3.InvokeRequired)
            {
                // 使用Invoke将操作排队到UI线程
                textBox3.Invoke(new Action(() => textBox3.Text = nSum.ToString()));
            }
            else
            {
                // 直接更新控件
                textBox3.Text = nSum.ToString();
            }


            return 0;
        }

        private void button_Calculate_Click(object sender, EventArgs e)
        {
            int a = 0,b = 0;
            Int32.TryParse(textBox1.Text,out a);
            Int32.TryParse(textBox2.Text,out b);
            AddNum(a,b, CallBackAddNum);
        }
    }
}

代码说明:

public delegate int CallBackFunc(int a, int b);
CallBackFunc callback = null;

1.上述代码使用delegate声明一个委托,并且将其初始化为null,在构造函数中将回调函数和委托进行绑定。

2.然后再按钮中调用导入的C++接口。

3.在回调函数中将计算结果更新到控件中。

这里有多种方式:

方式1:使用InvokeBeginInvoke方法

InvokeBeginInvoke是.NET提供的用于跨线程更新控件的方法。它们可以确保控件的更新操作在UI线程中执行。

示例代码:

 // 检查是否需要跨线程操作
            if (textBox3.InvokeRequired)
            {
                // 使用Invoke将操作排队到UI线程
                textBox3.Invoke(new Action(() => textBox3.Text = nSum.ToString()));
            }
            else
            {
                // 直接更新控件
                textBox3.Text = nSum.ToString();
            }

上述代码中

  • InvokeRequired:用于检查当前线程是否是控件的创建线程。如果是,则不需要跨线程操作;如果不是,则需要使用InvokeBeginInvoke
  • Invoke:同步执行委托,阻塞当前线程,直到操作完成。
  • BeginInvoke:异步执行委托,不会阻塞当前线程。

方式2:使用SynchronizationContext

SynchronizationContext是一个更通用的线程同步机制,可以用于在任何线程中同步到UI线程。

示例代码:

// 在主线程中保存SynchronizationContext
private SynchronizationContext _uiContext;

public Form1()
{
    InitializeComponent();
    _uiContext = SynchronizationContext.Current; // 保存主线程的SynchronizationContext
}

// 在委托回调函数中使用SynchronizationContext更新控件
public void UpdateButtonCallback()
{
    _uiContext.Post(_ =>
    {
        button1.Text = "更新后的文本";
    }, null);
}
  • SynchronizationContext.Current:获取当前线程的同步上下文。
  • Post:异步执行委托。

方式3:使用TaskContinueWith(异步任务)

如果你在异步任务中需要更新UI控件,可以使用ContinueWith方法,并指定在UI线程中执行。

示例代码:

// 假设有一个按钮控件button1和一个异步任务
public void UpdateButtonAsync()
{
    Task.Run(() =>
    {
        // 模拟耗时操作
        Thread.Sleep(2000);
    }).ContinueWith(t =>
    {
        button1.Text = "更新后的文本";
    }, TaskScheduler.FromCurrentSynchronizationContext()); // 确保在UI线程中执行
}
  • TaskScheduler.FromCurrentSynchronizationContext:指定任务继续在UI线程中执行。

总结

  • 如果是WinForms应用程序,推荐使用InvokeBeginInvoke
  • 如果是WPF应用程序,推荐使用Dispatcher.Invoke
  • 如果需要更通用的解决方案,可以使用SynchronizationContext
  • 如果是异步任务,可以使用TaskContinueWith

好了,关于在C#WinForm中使用委托实现回调函数功能的介绍就到这里了。
源码地址:Gitee

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/967754.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

从基础到人脸识别与目标检测

前言 从本文开始,我们将开始学习ROS机器视觉处理,刚开始先学习一部分外围的知识,为后续的人脸识别、目标跟踪和YOLOV5目标检测做准备工作。我采用的笔记本是联想拯救者游戏本,系统采用Ubuntu20.04,ROS采用noetic。 颜…

未来替代手机的产品,而非手机的本身

替代手机的产品包括以下几种: 可穿戴设备:智能手表、智能眼镜等可穿戴设备可以提供类似手机的功能,如通话、信息推送、浏览网页等。 虚拟现实(VR)技术:通过佩戴VR头显,用户可以进行语音通话、发…

QTreeView和QTableView单元格添加超链接

QTreeView和QTableView单元格添加超链接的方法类似,本文仅以QTreeView为例。 在QTableView仿Excel表头排序和筛选中已经实现了超链接的添加,但是需要借助delegate,这里介绍一种更简单的方式,无需借助delegate。 一.效果 二.实现 QHTreeView.h #ifndef QHTREEVIEW_H #def…

正则引入store中的modules文件

正则引入store中的modules文件 // index.js import { createStore } from vuex;const modulesFiles require.context(./modules, true, /\.ts|js$/); const modules modulesFiles.keys().reduce((modules1, modulePath) > {const moduleName modulePath.replace(/^\.\/(.…

如何保证Redis和MySQL数据的一致性刨析

1、常见的缓存更新策略: 定义:主要用来进行redis和mysql的数据同步更新的一些策略 内存淘汰:等触发淘汰机制后,刚好淘汰到了用户查询的数据,此时是null,会进行查询数据库并写入到缓存中,此时…

产品详情页中 品牌官网详情 对应后端的字段是 detail

文章目录 1、在这个Vue代码中,品牌官网详情 对应后端的字段是 detail2、品牌官网详情 功能相关的代码片段3、export const productSave (data: any) >4、ProductController5、ProductDto 类6、ProductApiService 1、在这个Vue代码中,品牌官网详情 对…

使用C语言实现MySQL数据库的增删改查操作指南

使用C语言与MySQL数据库进行交互,通常涉及使用MySQL提供的C API库。这套API允许开发者在C/C++程序中执行SQL查询,从而实现数据库的增删改查操作。下面,我将详细介绍如何在C语言中实现这些基本操作。 准备工作 安装MySQL开发库:确保你的系统上安装了MySQL服务器以及MySQL开发…

【蓝桥杯嵌入式】2_LED

全部代码网盘自取 链接:https://pan.baidu.com/s/1PX2NCQxnADxYBQx5CsOgPA?pwd3ii2 提取码:3ii2 1、电路图 74HC573是八位锁存器,当控制端LE脚为高电平时,芯片“导通”,LE为低电平时芯片“截止”即将输出状态“锁存”…

计算机视觉常用数据集Cityscapes的介绍、下载、转为YOLO格式进行训练

我在寻找Cityscapes数据集的时候花了一番功夫,因为官网下载需要用公司或学校邮箱邮箱注册账号,等待审核通过后才能进行下载数据集。并且一开始我也并不了解Cityscapes的格式和内容是什么样的,现在我弄明白后写下这篇文章,用于记录…

MariaDB MaxScale实现mysql8主从同步读写分离

一、MaxScale基本介绍 MaxScale是maridb开发的一个mysql数据中间件,其配置简单,能够实现读写分离,并且可以根据主从状态实现写库的自动切换,对多个从服务器能实现负载均衡。 二、MaxScale实验环境 中间件192.168.121.51MaxScale…

Python设计模式 - 原型模式

定义 原型模式是一种创建型设计模式,它可以通过复制现有对象来创建新对象,而不是直接实例化新的对象。 结构 抽象原型(Prototype):声明 clone() 方法,以便派生类实现克隆自身的能力。具体原型&#xff08…

GWO优化决策树回归预测matlab

灰狼优化算法(Grey Wolf Optimizer,简称 GWO)是一种群智能优化算法,由澳大利亚格里菲斯大学的 Mirjalii 等人于 2014 年提出。该算法的设计灵感源自灰狼群体的捕食行为,核心思想是模仿灰狼社会的结构与行为模式。 在本…

Oracle的学习心得和知识总结(三十三)|Oracle数据库数据库的SQL ID的底层计算原理分析

目录结构 注:提前言明 本文借鉴了以下博主、书籍或网站的内容,其列表如下: 1、参考书籍:《Oracle Database SQL Language Reference》 2、参考书籍:《PostgreSQL中文手册》 3、EDB Postgres Advanced Server User Gui…

Git(分布式版本控制系统)系统学习笔记【并利用腾讯云的CODING和Windows上的Git工具来实操】

Git的概要介绍 1️⃣ Git 是什么? Git 是一个 分布式版本控制系统(DVCS),用于跟踪代码的变更、协作开发和管理项目历史。 由 Linus Torvalds(Linux 之父)在 2005 年开发,主要用于 代码管理。…

yum报错 Could not resolve host: mirrorlist.centos.org

检查dns 使用ping www.baidu.com ,如果ping不通,检查/etc/resolv.conf文件中是否有: nameserver 8.8.8.8 nameserver 8.8.4.4 替换yum源 1.备份原始的 YUM 源配置文件: sudo cp /etc/yum.repos.d/CentOS-Base.repo /etc/yum.r…

postgreSQL16.6源码安装

1.获取源码 从PostgreSQL: File Browser获取tar.bz2或者tar.gz源码 2.解压 tar xf postgresql-version.tar.bz2 roothwz-VMware-Virtual-Platform:/usr/local# tar xf postgresql-16.6.tar.bz2 roothwz-VMware-Virtual-Platform:/usr/local# ll 总计 24324 drwxr-xr-x 12 ro…

Machine Learning:Introduction

文章目录 Machine LearningTrainingStep 1.Contract Function with Unknown ParametersStep 2.Define Loss from Training DataStep 3.Optimization Linear ModelPiecewise Linear CurveBeyond Piecewise Liner?FunctionLossOptimization Model Deformation Machine Learning …

【Java】多线程和高并发编程(三):锁(下)深入ReentrantReadWriteLock

文章目录 4、深入ReentrantReadWriteLock4.1 为什么要出现读写锁4.2 读写锁的实现原理4.3 写锁分析4.3.1 写锁加锁流程概述4.3.2 写锁加锁源码分析4.3.3 写锁释放锁流程概述&释放锁源码 4.4 读锁分析4.4.1 读锁加锁流程概述4.4.1.1 基础读锁流程4.4.1.2 读锁重入流程4.4.1.…

使用redis实现 令牌桶算法 漏桶算法

流量控制算法,用于限制请求的速率。 可以应对缓存雪崩 令牌桶算法 核心思想是: 有一个固定容量的桶,里面存放着令牌(token)。每过一定时间(如 1 秒),桶中会自动增加一定数量的令牌…

金媒婚恋交友系统V10.5的CRM操作提示:“您没有权限执行此操作”解决方法

大家都知道新年2.5日新版10.5已经升级了,这次升级相对以前更新内容相当重量级!最突出的就是CRM系统的更新和UI改观吐槽的内容都改进了我愿意和大家分享代码和新得~关注我昵称就能知道我哦!! 出现原因:是这个红娘账号没…