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

接上一篇,C++中,如何设计等价于Qt的信号与槽机制。

看下面例子:


class FileManager : public QObject {Q_OBJECTpublic:FileManager(QObject* parent = nullptr) : QObject(parent) {}void changeFileName(const QString& newName) {fileName = newName;emit fileNameChanged(fileName);}signals:void fileNameChanged(const QString& newName);private:QString fileName;
};

所有连接类FileManage中的信号,在changeFileName函数中被调用。例如:

int main(int argc, char* argv[])
{QApplication a(argc, argv);  //注意,这里是QApplicationFileManager fm;QObject::connect(&fm, &FileManager::fileNameChanged, [](const QString& newName) {std::cout << "函数1收到:文件名改变了。\n";});QObject::connect(&fm, &FileManager::fileNameChanged, [](const QString& newName) {std::cout << "函数2收到:文件名改变2。\n";});fm.changeFileName("abc");return a.exec();
}

输出:

下面我们也来设计相同功能的FileManager2:

class FileManager2 {
public:using fileNameChanged = std::function<void(const QString& newName)>;public:std::list< fileNameChanged> fileNameChanged_list;
public:FileManager2(QObject* parent = nullptr) {}void changeFileName(const QString& newName) {fileName = newName;for (fileNameChanged&  f: fileNameChanged_list) {if (f != 0)f(newName);}}void connect(fileNameChanged f) {fileNameChanged_list.push_back(f);}
private:QString fileName;
};int main(int argc, char* argv[])
{QApplication a(argc, argv);  //注意,这里是QApplicationFileManager2 fm;fm.connect([](const QString& newName) {std::cout << "函数1收到:文件名改变了。\n";});fm.connect([](const QString& newName) {std::cout << "函数2收到:文件名改变。\n";});fm.changeFileName("abc");return a.exec();
}

输出:

是不是一样,是不是很酷。:)

关键地方:

这里相当于Qt中的emit

这里添加一个函数指针,上篇说过:

这里可连接N个Lambda 表达式,与Qt一样:

连接函数:

最终也一样:

由于临时灵感来了,用了几分钟写的,难免有问题, 如果使用
std::list::remove 会编译不了,现在用新版方案替代,以后再完善:

template<class T>
class _callback {
public:friend bool  operator==(const _callback& l, const _callback& r) {return l.m_id == r.m_id;}_callback(const T& pf) {m_pf = pf;m_id = _DateTime::g_timeStamp();}public:__int64 m_id;T m_pf;
};class _FileManager {
public:using pf_fileNameChanged = std::function<void(const _string& oldPathName,const _string& newPathName)>;/// <summary>/// 更改文件名或路径并通知所有订阅者/// </summary>/// <param name="oldPathName"></param>/// <param name="newPathName"></param>/// 创建时间:2025-03-16    最后一次修改时间:2025-03-16void changeFileName(const _string& oldPathName, const _string& newPathName);/// <summary>/// 更改文件夹名或路径/// </summary>/// <param name="oldPathName"></param>/// <param name="newPathName"></param>/// 创建时间:2025-03-16    最后一次修改时间:2025-03-16void changeFolderName(const _string& oldPathName, const _string& newPathName);/// <summary>/// 删除文件/// </summary>/// <param name="fileName"></param>void deleteFile(const _string& fileName);/// <summary>/// 删除文件夹/// </summary>/// <param name="folderName"></param>void deleteFolder(const _string& folderName);// 订阅文件名更改事件void subscribe(const pf_fileNameChanged& callback) {std::lock_guard<std::mutex> lock(m_mutex); // 线程安全m_fileNameChanged.push_back(callback);}// 取消订阅文件名更改事件void unsubscribe(const pf_fileNameChanged& callback) {std::lock_guard<std::mutex> lock(m_mutex); // 线程安全m_fileNameChanged.remove(callback);}private:_string m_dirPathName; // 当前文件名std::list<_callback<pf_fileNameChanged>> m_fileNameChanged; // 订阅者列表mutable std::mutex m_mutex; // 用于线程安全
};

相关文章:

接上一篇,C++中,如何设计等价于Qt的信号与槽机制。

看下面例子&#xff1a; class FileManager : public QObject {Q_OBJECTpublic:FileManager(QObject* parent nullptr) : QObject(parent) {}void changeFileName(const QString& newName) {fileName newName;emit fileNameChanged(fileName);}signals:void fileNameChan…...

Spring(6)——Spring、Spring Boot 与 Spring MVC 的关系与区别

Spring、Spring Boot 与 Spring MVC 的关系与区别 1. 核心定位 Spring 定位&#xff1a;基础框架&#xff0c;提供 IoC&#xff08;控制反转&#xff09; 和 DI&#xff08;依赖注入&#xff09; 核心功能&#xff0c;管理对象生命周期及依赖关系。功能&#xff1a;支持事务管…...

安装baselines出现的环境配置问题

该错误通常是由于环境配置问题、依赖包缺失、权限不足等原因导致 1. 更新相关工具 pip install --upgrade pip setuptools 2. 检查并安装依赖 conda install setuptools pip wheel 出现新问题&#xff1a; 3.尝试使用 Conda 安装 conda install mpi4py 再尝试安装 baseli…...

【商城实战(38)】Spring Boot:从本地事务到分布式事务,商城数据一致性的守护之旅

【商城实战】专栏重磅来袭&#xff01;这是一份专为开发者与电商从业者打造的超详细指南。从项目基础搭建&#xff0c;运用 uniapp、Element Plus、SpringBoot 搭建商城框架&#xff0c;到用户、商品、订单等核心模块开发&#xff0c;再到性能优化、安全加固、多端适配&#xf…...

当今前沿技术:人工智能与区块链的未来发展

在如今快速发展的科技时代&#xff0c;各种前沿技术正在改变的生活。人工智能AI&#xff09;就是其中之一。它在医疗、金融、制造等多个领域发挥着巨大作用。AI可以分析数据&#xff0c;识别模式&#xff0c;还能辅助决策。比如&#xff0c;在医疗方面&#xff0c;AI帮助医生更…...

perl的package中“Subroutine new redefined”问题

我在一个脚本run_PMseq.V8.pl调用了一些.pm文件 $perl -c run_PMseq.V8.pl Subroutine new redefined at /mnt/lustre/user/wubin/01.Program/Scripts/01.script/GeneLab/PMSeq/package_V3/Add_mismatch.pm line 25. Subroutine generate_shell redefined at /mnt/lustre/use…...

markdown 转 word 工具 ‌Pandoc‌

‌Pandoc‌是一个开源的文档转换工具&#xff0c;由John MacFarlane开发&#xff0c;旨在提供一个通用的文档转换解决方案。它支持多种输入和输出格式&#xff0c;能够高效地将不同格式的文档进行转换‌ 功能 Pandoc支持以下格式之间的转换&#xff1a; **Markdown、reStruct…...

英语学习(GitHub学到的分享)

【英语语法&#xff1a;https://github.com/hzpt-inet-club/english-note】 【离谱的英语学习指南&#xff1a;https://github.com/byoungd/English-level-up-tips/tree/master】 【很喜欢文中的一句话&#xff1a;如果我轻轻松松的学习&#xff0c;生活的幸福指数会提高很多…...

【eNSP实战】三层交换机使用ACL实现网络安全

拓图 要求&#xff1a; vlan1可以访问Internetvlan2和vlan3不能访问Internet和vlan1vlan2和vlan3之间可以互相访问PC配置如图所示&#xff0c;这里不展示 LSW1接口vlan配置 vlan batch 10 20 30 # interface Vlanif1ip address 192.168.40.2 255.255.255.0 # interface Vla…...

Javascript BOM,DOM 知识简介

JSON 一种数据交换格式,作为数据载体,传输数据, Json比xml 更简单,可读性更高.js的对象和Json可以相互转换. //json定义格式: var varName{"key1":value1,"key2":value2};value的数据类型为数字,字符串(在双引号中),布尔值,数组(在方括号中),对象(在花括…...

拆解 “ES 已死“ 伪命题:Agentic RAG 时代搜索引擎的终极形态

作者&#xff1a;来自 Elastic 李捷 xxx&#xff1a;“ES已死&#xff0c;#%#……” 我&#xff1a;&#xff1f;&#xff1f;&#xff1f; 最近&#xff0c;某厂商发了一堆公关文章&#xff0c;翻来覆去地炒作 “ES 已死”&#xff0c;“放弃 ES”。这哪是什么正经的技术文章&…...

关于ISP Pipeline LSC(镜头阴影校正)位置的一些想法

关于LSC校正的一些基本原理可以参考如下链接&#xff1a; ISP之LSC 【ISP】浅析Lens Shading ISP-镜头阴影校正&#xff08;LSC&#xff09; 这篇博文不打算讲具体的LSC校正原理。 主要是答复一位网友关于LSC校正在ISP Pipeline的问题。 网友问题如下&#xff1a; Rin_Cyn…...

Vue学习笔记集--六大指令

内容渲染指令 内容渲染指令用来辅助开发者渲染 DOM 元素的文本内容。常用的内容渲染指令有如下2 个&#xff1a; v-text&#xff08;类似innerText&#xff09; 使用语法&#xff1a;<p v-text"name">hello</p>&#xff0c;意思是将 name 值渲染到 p 标…...

.net 6程序在IIS中部署后点击IIS设置报错“执行此操作时出错”

.net 6写的程序&#xff0c;需要在Windows服务器的IIS中部署&#xff0c;由于是刚装的系统&#xff0c;先安装.net 6运行时&#xff0c;装了才发现没有IIS&#xff0c;于是又通过“添加角色和功能”添加与IIS相关的功能。安装完毕后&#xff0c;在IIS中添加网站&#xff0c;并将…...

《从零手写Linux Shell:详解进程控制、环境变量与内建命令实现 --- 持续更新》

承接上文Linux 进程的创建、终止、等待与程序替换保姆级讲解-CSDN博客&#xff0c;涉及所用到的代码&#xff0c;本文所绑定的资源就是上篇文章的主要代码。 完整代码在文章末尾 目录 1.实现编写代码输出一个命令行 a.如何获取自己的用户名&#xff0c;主机名&#xff0c;路径…...

【Go语言圣经2.4】

目标 理解 在 Go 中&#xff0c;赋值操作既包括最基本的形式&#xff08;左边一个变量&#xff0c;右边一个表达式&#xff09;&#xff0c;也包括复合赋值、元组赋值和隐式赋值。表达式求值的顺序、变量更新时的副作用以及如何处理多返回值和下划线&#xff08;_&#xff09…...

运维工具推荐 -- 宝塔面板:一键部署服务器

标题&#xff1a;宝塔面板&#xff1a;一键部署服务器&#xff0c;轻松管理你的云端世界 引言 在数字化时代&#xff0c;服务器管理对于个人开发者、中小企业或站长来说既是机遇也是挑战。手动配置服务器环境耗时费力&#xff0c;而 宝塔面板 作为一款 免费开源、功能全面 的服…...

C# 异常处理‌的核心概念

文章目录 一、异常处理的核心概念‌‌二、C# 异常处理的基本语法‌‌三、常见异常类型‌‌四、最佳实践‌‌五、示例&#xff1a;文件读取异常处理‌‌六、总结‌ C# 异常处理‌的详细说明&#xff0c;包括核心概念、使用方法和最佳实践&#xff1a; 一、异常处理的核心概念‌ …...

腾讯云点播key防盗链生成到期自动失效url

package com.xmkjsoft.protect_key;import java.nio.charset.StandardCharsets; import java.security.MessageDigest;public class TencentKeyAntiTheft {private static final String SECRET_KEY ""; // 请替换为腾讯云 VOD 控制台中的 Key/*** 生成腾讯云 Key 防…...

深入 Spring Boot 注解

深入 Spring Boot 注解&#xff1a;我的开发心得与常用注解详解 大家好&#xff0c;我是 [你的 CSDN 昵称/名字]&#xff0c;一位热爱 Spring Boot 的技术博主。 在多年的 Spring Boot 开发实践中&#xff0c;我深深体会到注解的强大魅力。它们不仅让代码变得更简洁&#xff0…...

k8s环境部署

四台机器 分别是 k8s-master&#xff1a;172.25.254.100 k8s-node1&#xff1a;172.25.254.10 k8s-node2&#xff1a;172.25.254.20 docker-harbor&#xff1a;172.25.254.200 reg.timinglee.org 四台机器分别配置好网络和软件仓库 做好地址解析 scp -r /etc/hosts/ root17…...

CentOS 系统安装 docker 以及常用插件

博主用的的是WindTerm软件链接的服务器&#xff0c;因为好用 1.链接上服务器登入后&#xff0c;在/root/目录下 2.执行以下命令安装docker sudo yum install -y yum-utilssudo yum-config-manager \--add-repo \https://download.docker.com/linux/centos/docker-ce.reposudo…...

谷歌云服务器:服务器怎么安装???

谷歌云服务器&#xff1a;服务器怎么安装&#xff1f;&#xff1f;&#xff1f; 以下是详细分步指南&#xff0c;帮助你在 Google Cloud Platform (GCP) 上快速创建并配置云服务器&#xff08;Compute Engine 实例&#xff09;&#xff0c;并安装所需环境&#xff1a; 一、准备…...

Redis--Zset类型

目录 一、引言 二、介绍 三、命令 1.zadd 2.zrange&#xff0c;zrevrange&#xff0c;zrangebyscore 3.zcard&#xff0c;zcount 4.zpopmax&#xff0c;bzpopmax&#xff0c;zpopmin&#xff0c;bzpopmin 5.zrank,zrevrank,zscore 6.zrem&#xff0c;zremrangebyrank&a…...

《阿里云Data+AI:开启数据智能新时代》电子书上线啦!

本书整理了阿里云在DataAI领域的最新实践案例与深度洞察&#xff0c;涵盖电商、游戏、营销、数字内容等多个行业的成功经验&#xff0c;以及技术专家对数据库与AI融合趋势的专业解读。 通过理论与实践的结合&#xff0c;我们将共同探索DataAI如何成为企业智能化转型的核心驱动…...

图像处理篇---图像预处理

文章目录 前言一、通用目的1.1 数据标准化目的实现 1.2 噪声抑制目的实现高斯滤波中值滤波双边滤波 1.3 尺寸统一化目的实现 1.4 数据增强目的实现 1.5 特征增强目的实现&#xff1a;边缘检测直方图均衡化锐化 二、分领域预处理2.1 传统机器学习&#xff08;如SVM、随机森林&am…...

Vue 3 事件总线详解:构建组件间高效通信的桥梁

Vue 3 事件总线详解&#xff1a;构建组件间高效通信的桥梁 为什么需要事件总线&#xff1f;使用 mitt 实现事件总线1. 安装 mitt2. 创建事件总线3. 在组件中使用事件总线发送端组件&#xff08;例如 ComponentA.vue&#xff09;接收端组件&#xff08;例如 ComponentB.vue&…...

Golang编译器DIY,手搓 if err != nil { return err } 语法糖

前序 在go的社区里&#xff0c;下面这三行代码是被吐槽的最多的 if err ! nil {return err }从代码之整洁美观的角度看&#xff0c;这样的写法也是让人不舒服的。尤其是 当有很多错误需要处理的时候&#xff0c;就会发现通篇都是这三行。 所以想着看看修改一下编译器&#xf…...

图解多头注意力机制:维度变化一镜到底

目录 一、多头注意力机制概述二、代码实现1. pyTorch 实现2. tensorFlow实现 三、维度变化全流程详解1. 参数设定2. 维度变化流程图3. 关键步骤维度变化 四、关键实现细节解析1. 多头拆分与合并2. 注意力分数计算3. 掩码处理技巧 五、完整运行示例六、总结与常见问题1. 核心优势…...

.NET_Prism基本项目创建

Prism简述 模块&#xff08;Module&#xff09;&#xff1a;独立的功能单元&#xff0c;可动态加载。依赖注入&#xff08;Dependency Injection&#xff0c;DI&#xff09;&#xff1a;通过 IoC 容器&#xff08;如 Unity、Autofac&#xff09;管理模块之间的依赖关系。&#…...