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

【Qt中信号槽连接connect有接收者和无接收者的区别】

Qt中信号槽连接connect有接收者和无接收者的区别

    • 概述
    • 代码验证
    • 结论

概述

在qt开发过程中,在多线程的环境下,发现程序有些问题,根据程序打印出来的提示,发现与线程有关。最终定位到connect函数处,于是有以下的研究:关于connect函数有接收者和无接收者的区别。

代码验证

验证工程代码文件包括:define.h(类的定义) mainwindow类文件。
根据打印结果(可以看mainwindow.cpp的注释),可以得出结论(也在注释里面)
注意:尽量使用有接收者的connect函数,同时,信号方的线程指的是发送信号的线程,而不是信号拥有者的线程。
define.h:

#ifndef DEFINE_H
#define DEFINE_H
#include<QDebug>
#include<QDateTime>
#pragma execution_character_set("utf-8") //如果源文件是UTF-8+无BOM的编码方式,则一定不能加#pragma execution_character_set(“utf-8”)
#define myDebug qDebug()<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss")<<__FILE__<<__LINE__
#include<QThread>
class Worker: public QObject
{Q_OBJECT
public:Worker(QObject* parent = nullptr): QObject(parent) {}signals:void finished();public slots:void work(){myDebug << "工作完成 worker";myDebug << QThread::currentThread() << " ==" << sender()->thread() << "==" << this->thread() << this->metaObject()->className();emit finished();}
};class Worker2: public QObject
{Q_OBJECT
public:Worker2(QObject*parent = nullptr): QObject(parent){}void emitAfterWork(){emit afterWork();}
signals:void afterWork();
public slots:void finishedSlots(){myDebug << "工作完成,下班,Work2";myDebug << QThread::currentThread() << " ==" << sender()->thread() << "==" << this->thread() << this->metaObject()->className();emit afterWork();}
public:};
#endif // DEFINE_H

mainwindow.h:

#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include"define.h"
QT_BEGIN_NAMESPACE
namespace Ui
{class MainWindow;
}
QT_END_NAMESPACEclass MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr);~MainWindow();private:Ui::MainWindow* ui;Worker* wk;Worker2* wk2;
};
#endif // MAINWINDOW_H

mainwindow.cpp:

#include "mainwindow.h"
#include "ui_mainwindow.h"MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);myDebug;wk2 = new Worker2;wk = new Worker;QThread *thread = new QThread;wk->moveToThread(thread);thread->start();connect(ui->pushButton, &QPushButton::released, wk, &Worker::work);//按钮会触发Worker类的finished信号,且是在Worker所在线程的。//wk与当前类对象不在同一线程:wk在子线程,而当前类对象在主线程。//无接收者的connect,槽函数的执行线程是由信号端决定的,打印结果:QThread(0x6c80c8)  == QThread(0x6c80c8) == QThread(0x6c5f50) MainWindow//结果表明,在没有接收者的情况下,槽函数所在线程就是信号端的线程(至于是发送者的线程  还是  执行发送信号的线程要从下面另一个connect确定)//有接收者的connect,槽函数与接收者在同一个线程,打印结果:QThread(0x13ed680)  == QThread(0x13f8748) == QThread(0x13ed680) MainWindow//结果表明,在接收者的情况下,槽函数所在线程就是接收者的线程connect(wk, &Worker::finished, /*this,*/ [ = ](){myDebug << "工作完成,下班,Work2";myDebug << QThread::currentThread() << " ==" << wk->thread() << "==" << this->thread() << this->metaObject()->className();wk2->emitAfterWork();});//此处connect用于判断发送端的线程是:发送者的线程 还是 发送信号的线程//由于wk2与当前类都在主线程里面,而“上面的connect使用无接收者的connect函数,则信号会在子线程发出”//如果发送端的线程是指wk2所在的线程,那么下面connect无论是有接收者还是没有,那打印出来的线程应该都是一样的。//如果不一样,则说明线程由发送信号的线程决定而不是发送者所在的线程//无接收者connect打印结果:QThread(0x767720)  == QThread(0x7657d8) == QThread(0x7657d8) MainWindow//结果表明:在没有接收者的情况下,槽函数所在线程是由信号端确定的,但这里线程又不是发送者所在的线程,那么就是槽函数的线程就是由发送信号的线程所确定//有接收者connect打印结果:QThread(0x1316550)  == QThread(0x1316550) == QThread(0x1316550) MainWindow//总结,无论什么情况都应该尽量使用有接收者的connect,而无接收者的connect 大致等价与 有接收者的connect的第五个参数为Qt::DirectConnection。//即槽函数在发送信号的线程中执行,这样是不安全的。connect(wk2, &Worker2::afterWork, this, [ = ](){myDebug << "收到下班信号,mainwindow";myDebug << QThread::currentThread() << " ==" << wk2->thread() << "==" << this->thread() << this->metaObject()->className();});}MainWindow::~MainWindow()
{delete ui;
}

结论

总结,无论什么情况都应该尽量使用有接收者的connect,而无接收者的connect 大致等价与 有接收者的connect的第五个参数为Qt::DirectConnection。 即槽函数在发送信号的线程中执行,这样是不安全的。

http://www.xdnf.cn/news/19724.html

相关文章:

  • 执行一条select语句期间发生了什么?
  • 常用符号 Emoji 对照表——Unicode UTF-8
  • CSS Sass Less 样式.xxx讲解
  • SpringMVC的请求接收与结果响应
  • 华为HCIE数通含金量所剩无几?考试难度加大?
  • 数据库选择有讲究?SQLite、PostgreSQL还是MySQL?
  • 电脑接入企业中的网线,为啥网卡上面显示AD域名
  • MongoDB 聚合查询超时:索引优化与分片策略的踩坑记录
  • 国产CAD皇冠CAD(CrownCAD)建模教程:汽车驱动桥
  • 二、Scala流程控制:分支与循环
  • 波浪模型SWAN学习(2)——波浪浅化模拟(Shoaling on sloping beach)
  • RoPE频率缩放机制:解密大语言模型上下文扩展的核心算法
  • linux之IO存储子系统全流程分析
  • 差分隐私在运营指标:ABP 的 DP 计数器与噪声预算
  • 使用PyTorch构建全连接神经网络实现MNIST手写数字分类
  • 【面试题】 如何处理中文分词?
  • LeetCode 2486.追加字符以获得子序列
  • ubuntu的2T新硬盘分区、格式化并挂载
  • Python进阶第三方库之Numpy
  • GO : cannot find module
  • 【音视频】 RGB 格式详解
  • 1.Linux:命令提示符,history和常用快捷键
  • 程序员之电工基础-初尝线扫相机
  • 百度发布Comate AI IDE,我要把Cursor卸载了!
  • AI生成PPT工具排名:2025年高效办公新选择
  • 【项目】分布式Json-RPC框架 - 应用层实现
  • Docker 安装 RAGFlow保姆教程
  • 【大前端】React 使用 Redux 实现组件通信的 Demo 示例
  • Vue 评论组件设计 V1.0
  • JVM 的 C1/C2 编译器