Qt 的信号槽机制详解:之信号槽引发的 Segmentation Fault 问题拆析(上)
Qt 的信号槽机制详解:之因信号槽误用引发的 Segmentation Fault 问题拆析(上)
- 前言
- 一. 信号与槽的基本概念
- 信号(Signal)
- 槽(Slot)
- 连接信号与槽
- 二. 信号槽机制的实现原理
- 元对象系统(Meta-Object System)
- 信号与槽的内部数据结构
- 三. 信号与槽的连接方式
- 经典语法
- 新语法(推荐)
- Lambda 表达式
- 静态槽
- 四. 信号与槽的连接类型
- 1. Qt::AutoConnection(默认)
- 2. Qt::DirectConnection
- 3. Qt::QueuedConnection
- 4. Qt::BlockingQueuedConnection
- 5. Qt::UniqueConnection
- 五. 信号与槽的高级特性
- 1. 多对多连接
- 2. 可断开连接
- 3. 自定义信号与槽
- 六. 信号槽的线程安全性
- 七. 性能分析
- 总结
前言
该系列文章中,我主要和大家一同探讨因为Qt 的信号槽机制误用,而引发的 Segmentation Fault 问题。
作为自己目前经手项目的阶段性总结,同时也给大家分享几个正确使用QT信号槽的定向性方式。
在该篇内容中,我将结合 Qt 编译器的特性,详细分析信号槽使用不当可能引发的崩溃问题。
【系列文章】索引:
Qt 的信号槽机制详解:之信号槽引发的 Segmentation Fault 问题拆析(上)
Qt 的信号槽机制详解:之信号槽引发的 Segmentation Fault 问题拆析(下)
Qt 的信号与槽(Signal & Slot)机制是其核心特性之一,用于实现组件之间的解耦通信。
它本质上是基于 观察者模式(Observer Pattern) 的一种实现,但由于 Qt 信号槽系统结合了 元对象系统(Meta-Object System),使其更加高效和灵活。
一. 信号与槽的基本概念
信号(Signal)
- 信号是一个由对象发出的通知,表示某个事件的发生。
- 信号没有返回值,但可以携带参数。
- 信号是由
emit关键字触发的。 - 信号是
protected或public的类成员,不能直接调用,只能通过emit触发。
槽(Slot)
- 槽是一个函数,用于处理信号发出的通知。
- 槽可以是类的普通成员函数,也可以是 lambda 表达式。
- 一个槽函数可以连接到多个信号。
- 槽函数也可以是静态函数或全局函数,但这需要通过适配器(如
std::bind或 lambda)来实现。
连接信号与槽
- 通过
QObject::connect将信号与槽连接。 - 当信号被触发时,Qt 自动调用与之连接的槽函数。
二. 信号槽机制的实现原理
元对象系统(Meta-Object System)
Qt 的信号槽机制依赖于元对象系统(QObject 和 QMetaObject)。
- 每个继承自
QObject的类都会通过moc工具生成对应的元对象代码。 - 元对象包含类的元数据(包括信号、槽、属性等信息)。
- 通过元对象系统,Qt 可以在运行时找到信号和槽,并在触发信号时动态调用槽函数。
信号与槽的内部数据结构
-
信号
- 信号在
QObject的子类中以QMetaObject的形式注册。 - 信号的发射会触发
QObject中的QMetaObject::activate()函数,该函数会查找所有连接的槽并调用它们。
- 信号在
-
槽
- 槽函数被存储为回调函数指针或函数对象(如 lambda)。
- 信号触发时,通过函数指针调用槽函数。
-
连接
- Qt 会在内部建立一个连接映射表,记录每个信号与对应槽的关系。
三. 信号与槽的连接方式
Qt 提供多种方式连接信号与槽:
经典语法
QObject::connect(sender, SIGNAL(signalName(params)), receiver, SLOT(slotName(params)));
- 缺点:使用字符串,容易导致参数错误,无法检查类型。
新语法(推荐)
QObject::connect(sender, &SenderClass::signalName, receiver, &ReceiverClass::slotName);
- 优点:类型安全,编译时检查参数匹配。
Lambda 表达式
QObject::connect(sender, &SenderClass::signalName, [](int value) {qDebug() << "Signal emitted with value:" << value;
});
- 优点:可以在槽中直接捕获局部变量。
静态槽
QObject::connect(sender, &SenderClass::signalName, staticFunction);
- 优点:适合不依赖实例的全局函数。
四. 信号与槽的连接类型
连接类型由 Qt::ConnectionType 决定:
1. Qt::AutoConnection(默认)
- 如果信号和槽在同一线程中,使用直接调用(Direct Connection)。
- 如果信号和槽跨线程,使用事件队列(Queued Connection)。
2. Qt::DirectConnection
- 信号触发时,直接调用槽函数(在信号的调用线程中执行)。
- 适用于需要实时响应的情况,但线程不安全。
3. Qt::QueuedConnection
- 信号触发时,将调用请求放入目标线程的事件队列中,由目标线程处理。
- 用于跨线程通信。
4. Qt::BlockingQueuedConnection
- 类似于
Qt::QueuedConnection,但信号触发线程会阻塞,直到槽函数执行完毕。 - 注意:容易引发死锁,应谨慎使用。
5. Qt::UniqueConnection
- 确保一个信号只连接到某个槽一次。
- 可以与其他类型联合使用,如
Qt::UniqueConnection | Qt::QueuedConnection。
五. 信号与槽的高级特性
1. 多对多连接
- 一个信号可以连接到多个槽。
- 一个槽可以连接到多个信号。
- 信号也可以连接到另一个信号。
示例:
QObject::connect(sender, &SenderClass::someSignal, receiver1, &ReceiverClass::slot1);
QObject::connect(sender, &SenderClass::someSignal, receiver2, &ReceiverClass::slot2);
QObject::connect(sender, &SenderClass::someSignal, sender, &SenderClass::anotherSignal);
2. 可断开连接
- 可以使用
QObject::disconnect断开信号与槽的连接:
QObject::disconnect(sender, &SenderClass::someSignal, receiver, &ReceiverClass::someSlot);
- 如果不指定槽,则断开所有与信号相关的连接。
3. 自定义信号与槽
- 可以在类中自定义信号与槽。
示例:
class MyClass : public QObject {Q_OBJECTsignals:void mySignal(int value);public slots:void mySlot(int value) {qDebug() << "Value received:" << value;}
};// 使用
MyClass obj;
QObject::connect(&obj, &MyClass::mySignal, &obj, &MyClass::mySlot);
emit obj.mySignal(42);
六. 信号槽的线程安全性
- 信号槽机制默认是线程安全的,尤其是在
Qt::QueuedConnection模式下。 - 直接连接(
Qt::DirectConnection)需要开发者手动确保线程安全。 - 在跨线程通信中,尽量使用
Qt::QueuedConnection。
七. 性能分析
信号槽的调用效率虽略低于直接调用,但优化得非常好:
- 直接连接的效率与函数指针调用接近。
- 使用事件队列的性能则取决于事件的处理速度。
- 如果性能是关键问题,可以使用 Qt 的低级机制(如函数指针)代替信号槽。
总结
Qt 的信号槽机制使得组件间的通信变得简单且高效,但也需要注意以下几点:
- 确保信号与槽的参数完全匹配。
- 合理选择连接类型,尤其是跨线程通信时。
- 使用
QObject::disconnect或对象生命周期管理机制,防止悬挂指针问题。 - 对于复杂场景,可结合 lambda 表达式或
std::bind使用。
相关文章:
Qt 的信号槽机制详解:之信号槽引发的 Segmentation Fault 问题拆析(上)
Qt 的信号槽机制详解:之因信号槽误用引发的 Segmentation Fault 问题拆析(上) 前言一. 信号与槽的基本概念信号(Signal)槽(Slot)连接信号与槽 二. 信号槽机制的实现原理元对象系统(M…...
uniapp中实现APP调用本地通知栏通知、震动、本地提示音或者mp3提醒
要在uniapp中实现APP调用本地通知栏通知、震动和本地提示音或者mp3提醒,你可以使用uni-app提供的原生API和插件来实现。 通知栏通知: 你可以使用uni-app的原生API uni.showToast() 或者 uni.showModal() 来实现通知栏通知的功能。可以在需要发送通知的地…...
ADB 上传文件并使用脚本监控上传百分比
有个需求,需要测试 emmc的外部连续写入性能,使用 ADB 上传一个巨大的文件。并且在上传到一定值时进行干预。 因此但是 adb push 命令本身会 block 运行并且不返回进度,因此需要一个额外的监控脚本。 上传脚本: echo off setloc…...
【数据库】PostgreSQL(持续更新)
目录 K8S 部署基本使用高级特性 K8S 部署 # pg.yaml --- apiVersion: v1 kind: PersistentVolume metadata:name: tv-postgres-pvnamespace: locallabels:storage: tv-postgres-pv spec:accessModes:- ReadWriteOncecapacity:storage: 50Gi # 按需修改,需要保持与…...
overleaf中出现TeX capacity exceeded PDF object stream buffer=5000000的原因和解决方案
在插入pdf 配图后,编译出错提示信息如图,很可能的一个原因是pdf文件大小太大了,最好压缩一下,压缩到1MB以内。...
pyqt5冻结+分页表
逻辑代码 # -*- coding: utf-8 -*- import sys,time,copy from PyQt5.QtWidgets import QWidget,QApplication, QDesktopWidget,QTableWidgetItem from QhTableWidgetQGN import Ui_QhTableWidgetQGN from PyQt5.QtCore import Qt from PyQt5 import QtCore, QtGui, QtWidgets…...
一起学Git【第四节:添加和提交文件】
通过前三节的学习,基本上对Git有了初步的了解,下面开始进行文件的添加和提交的流程。 这里主要涉及四个命令: git init 创建仓库git status查看仓库状态git add添加至暂存区git commit提交文件之前已经使用过git init命令了,此处不再具体讲解。参照一起学Git【第二节:创建…...
【鸿蒙实战开发】HarmonyOS集成高德地图定位实现
背景 随着HarmoneyOS 应用的井喷式增长,各大厂商也都加快了自己原生应用鸿蒙化的脚步,今天使用高德打车的时候忽然间想到高德在鸿蒙上有没有实现呢?打开next bate 版本的手机发现高德已经上架了,但是功能还不是特别完善。那么几乎每个应用都…...
ECharts散点图-气泡图,附视频讲解与代码下载
引言: ECharts散点图是一种常见的数据可视化图表类型,它通过在二维坐标系或其它坐标系中绘制散乱的点来展示数据之间的关系。本文将详细介绍如何使用ECharts库实现一个散点图,包括图表效果预览、视频讲解及代码下载,让你轻松掌握…...
python操作Elasticsearch执行增删改查
文章目录 基本操作更多查询方法1. 查询全部数据2. 针对某个确定的值/字符串的查询:term、match3. 在多个选项中有一个匹配,就查出来:terms4. 数值范围查询:range5. 多个条件同时触发 bool6. 指定返回值个数 size7. 返回指定列 _so…...
学习C++:关键字
关键字: 作用:关键字是C预先保留的单词(标识符) 在定义变量或者常量时候,不要用关键字 不要用关键字给变量或者常量起名称...
FFmpeg在python里推流被处理过的视频流
链式算法处理视频流 视频源是本地摄像头 # codinggbk # 本地摄像头直接推流到 RTMP 服务器 import cv2 import mediapipe as mp import subprocess as sp# 初始化 Mediapipe mp_drawing mp.solutions.drawing_utils mp_drawing_styles mp.solutions.drawing_styles mp_holis…...
为什么推荐使用构造函数注入而非@Autowired注解进行字段注入
在 Spring 框架中,推荐使用构造函数注入而非Autowired注解进行字段注入,主要有以下几个原因: 1. 依赖不可变和空指针安全 构造函数注入:使用构造函数注入时,依赖在对象创建时就必须提供,一旦对象创建完成&…...
如何卸载和升级 Angular-CLI ?
Angular-CLI 是开发人员使用 Angular 的必备工具。然而,随着频繁的更新和新版本的出现,了解如何有效地卸载和升级 Angular-CLI 对开发人员来说至关重要。本指南提供了一个全面的、循序渐进的方法来帮助您顺利过渡到最新版本。 必备条件 确保您的系统上…...
在线excel编辑(luckysheet)
项目地址:Luckysheet: 🚀Luckysheet ,一款纯前端类似excel的在线表格,功能强大、配置简单、完全开源。 可以下载项目使用npm安装运行,也可以用cdn 加载excel文件(使用luckyexcel): …...
【ES6复习笔记】Symbol 类型及其应用(9)
一、Symbol 简介 Symbol 是 JavaScript 中的一种基本数据类型,它表示唯一的标识符。Symbol 的主要目的是防止属性名冲突,尤其是在多个代码库或模块中共享对象时。Symbol 值可以用作对象的属性名,这样可以确保属性名是唯一的,不会…...
[原创](Modern C++)现代C++的第三方库的导入方式: 例如Visual Studio 2022导入GSL 4.1.0
[简介] 常用网名: 猪头三 出生日期: 1981.XX.XX 企鹅交流: 643439947 个人网站: 80x86汇编小站 编程生涯: 2001年~至今[共23年] 职业生涯: 21年 开发语言: C/C、80x86ASM、PHP、Perl、Objective-C、Object Pascal、C#、Python 开发工具: Visual Studio、Delphi、XCode、Eclipse…...
【ES6复习笔记】Class类(15)
介绍 ES6 提供了更接近传统语言的写法,引入了 Class(类)这个概念,作为对象的模板。通过 class 关键字,可以定义类。基本上,ES6 的 class 可以看作只是一个语法糖,它的绝大部分功能,…...
【Golang 面试题】每日 3 题(六)
✍个人博客:Pandaconda-CSDN博客 📣专栏地址:http://t.csdnimg.cn/UWz06 📚专栏简介:在这个专栏中,我将会分享 Golang 面试中常见的面试题给大家~ ❤️如果有收获的话,欢迎点赞👍收藏…...
openEuler安装OpenGauss5.0
OpenGauss5.0 Linux服务器 极简版 服务器安装 单节点安装 极简版安装 安装准备 获取安装包 下载地址:https://opengauss.org/zh/download/archive/版本选择:5.0.0 (LTS)系统架构:uname -m操作系统:cat /etc/os-release完整性校…...
Phi-4-mini-reasoning从零开始:学生自学AI推理工具搭建全过程
Phi-4-mini-reasoning从零开始:学生自学AI推理工具搭建全过程 1. 为什么选择Phi-4-mini-reasoning 作为一名对AI技术充满好奇的学生,我一直在寻找一个适合自学的推理模型。Phi-4-mini-reasoning以其专注推理任务的特点吸引了我。与通用聊天模型不同&am…...
5分钟搞定!FLUX.2-Klein-9B在ComfyUI中的快速部署与初体验
5分钟搞定!FLUX.2-Klein-9B在ComfyUI中的快速部署与初体验 1. 为什么选择FLUX.2-Klein-9B 如果你正在寻找一个既能高质量生成图像,又对中文提示词理解优秀的AI模型,FLUX.2-Klein-9B值得一试。这个模型特别适合需要频繁进行图像编辑的场景&a…...
Wan2.1视频生成创意玩法:把你的想法变成动态视觉故事
Wan2.1视频生成创意玩法:把你的想法变成动态视觉故事 1. 从文字到视频的魔法 你有没有过这样的经历?脑海中浮现出一个绝妙的创意场景,却苦于没有专业的视频制作技能将它呈现出来。或许是一个科幻故事的开场,一个产品演示的构想&…...
bert-base-chinese新手必看:完形填空与语义相似度功能实测教程
bert-base-chinese新手必看:完形填空与语义相似度功能实测教程 1. 快速了解bert-base-chinese bert-base-chinese是Google发布的经典中文预训练模型,作为NLP领域的基础模型,它已经成为中文自然语言处理任务的标准选择之一。这个模型特别适合…...
ChatGLM3-6B-128K vs ChatGLM3-6B:Ollama环境下的8K/128K场景选型指南
ChatGLM3-6B-128K vs ChatGLM3-6B:Ollama环境下的8K/128K场景选型指南 1. 两种模型的核心差异 ChatGLM3-6B和ChatGLM3-6B-128K都是基于ChatGLM3系列的开源对话模型,但它们在处理长文本能力上有着本质区别。 ChatGLM3-6B是标准版本,支持最多…...
Hunyuan-MT-7B性能实测:像素语言传送门在单卡A10上并发10路翻译的延迟与稳定性报告
Hunyuan-MT-7B性能实测:像素语言传送门在单卡A10上并发10路翻译的延迟与稳定性报告 1. 测试背景与目标 像素语言传送门(Pixel Language Portal)是基于腾讯Hunyuan-MT-7B模型构建的创新翻译工具,其独特的16-bit像素冒险界面设计为…...
OpenClaw浏览器自动化:千问3.5-27B驱动智能检索与内容聚合
OpenClaw浏览器自动化:千问3.5-27B驱动智能检索与内容聚合 1. 为什么需要浏览器自动化助手 作为一个经常需要做市场调研的技术人,我过去总是陷入这样的循环:打开十几个浏览器标签页,在不同平台间反复切换,手动复制粘…...
OpenClaw配置文件详解:Qwen3.5-9B高级参数调优手册
OpenClaw配置文件详解:Qwen3.5-9B高级参数调优手册 1. 为什么需要手动调优OpenClaw配置 上周我尝试用OpenClaw自动处理一批技术文档的归档工作,发现同样的任务在不同时段完成速度差异巨大。有时30分钟就能搞定,有时却要卡顿近2小时。这促使…...
用九齐单片机NY8B062F定时器实现精准延时与系统时基:从4ms中断到1秒计时的完整工程实践
九齐单片机NY8B062F定时器工程实战:构建高精度时基与延时系统 在嵌入式系统开发中,定时器如同设备的心跳,为各类功能提供精准的时间基准。九齐NY8B062F作为一款高性价比8位单片机,其四组灵活配置的定时器资源尤其适合小家电、智能…...
告别穿模与漂移!南洋理工团队提出HMR新框架:用视觉大模型对齐人体姿态
点击下方卡片,关注「3D视觉工坊」公众号选择星标,干货第一时间送达本文一作投稿发布 | 来源:3D视觉工坊「3D视觉从入门到精通」知识星球(点开有惊喜) !星球内有20多门3D视觉系统课程、300场顶会讲解、顶会论文最新解读、海量3D视觉…...
