qt中其他线程的信号向异步线程的槽传指针与释放指针问题

2023-05-16

发现qt中信号槽传指针的方式进行时,会发生释放不掉指针所分配的空间的情况。

定义一个信号

void emit_showTime(Data*);


void slot_showTime(Data*);

在发出信号之前动态创建一个Data的指针对象,并分配空间。传入信号中。然后在槽中进行delete 释放所分配的空间。


发现所传如的指针是正确的(地址都相同),但确不能如愿的释放掉该指针所分配的空间。使得程序占用内存不断增加。


有兴趣的可以试试,放在一个QTimer中不断执行。就能看出内存不断增加。

当信号与槽采用传值或引用时,却又是正确的。

定义一个信号

void emit_showTime(Data);


void slot_showTime(Data);

以此方式,则不会出现内存一直增加的问题。

不是一个好的做法。原因在于发送信号的地方很有可能是在对象的作用域范围,当信号发出后,槽函数立即被调用,而调用还没有返回的情况下,你却强行删除了对象,这样就存在一个潜在的致命错误。举个例子
void func{
Data data;
emit emit_showTime(&data); //这行意味着槽函数被马上调用并等待返回,如果在槽函数中删除data就会存在一个内存操作的危险
}
这类似于下面的操作
func{
Data data;
delete &data;
}
尽管你是使用动态创建对象,这也不是一个好的编程习惯,最好是这样
func{
Data *data = new Data;
emit emit_showTime(data);
delete data;
}

Qt的signal/slot机制原理

signal/slot在底层会使用三种方式传递消息。参见QObject::connect()方法:
bool QObject::connect ( const QObject * sender, const char * signal, const QObject * receiver, const char * method, Qt::ConnectionType type = Qt::AutoCompatConnection )
最后一个参数是就是传递消息的方式了,有四个取值:

Qt::DirectConnection
When emitted, the signal is immediately delivered to the slot.
假设当前有4个slot连接到QPushButton::clicked(bool),当按钮被按下时,QT就把这4个slot按连接的时间顺序调用一遍。显然这种方式不能跨线程(传递消息)。

Qt::QueuedConnection
When emitted, the signal is queued until the event loop is able to deliver it to the slot.
假设当前有4个slot连接到QPushButton::clicked(bool),当按钮被按下时,QT就把这个signal包装成一个 QEvent,放到消息队列里。QApplication::exec()或者线程的QThread::exec()会从消息队列里取消息,然后调用 signal关联的几个slot。这种方式既可以在线程内传递消息,也可以跨线程传递消息。

Qt::BlockingQueuedConnection
Same as QueuedConnection, except that the current thread blocks until the slot has been delivered. This connection type should only be used for receivers in a different thread. Note that misuse of this type can lead to dead locks in your application.
与Qt::QueuedConnection类似,但是会阻塞等到关联的slot都被执行。这里出现了阻塞这个词,说明它是专门用来多线程间传递消息的。

Qt::AutoConnection
If the signal is emitted from the thread in which the receiving object lives, the slot is invoked directly, as with Qt::DirectConnection; otherwise the signal is queued, as with Qt::QueuedConnection.
这种连接类型根据signal和slot是否在同一个线程里自动选择Qt::DirectConnection或Qt::QueuedConnection

这样看来,第一种类型的效率肯定比第二种高,毕竟第二种方式需要将消息存储到队列,而且可能会涉及到大对象的复制(考虑sig_produced(BigObject bo),bo需要复制到队列里)。
 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

qt中其他线程的信号向异步线程的槽传指针与释放指针问题 的相关文章

  • Windows11装新环境问题

    系统重装 1 下载最新的老毛桃U盘启动盘制作工具 xff08 旧版的会不支持windows11启动项 xff09 xff0c 制作U盘启动盘 2 将电脑的bitLocker硬盘加密取消掉在所有设置安全加密里 xff0c 否则将无法启动提示硬
  • c#引用office组件库迁移源码问题

    1 当程序引用了office组件时 xff0c 需要安装与迁移的源码相同版本的offce Library才行 xff0c 否则会报异常 2 可以下载office三合一版快捷安装 xff0c vs在引用里的com选项卡里会自动识别到相应的of
  • TCP和UDP的发送缓冲区和接收缓冲区内存问题

    TCP协议是作用是用来进行端对端数据传送的 xff0c 那么就会有发送端和接收端 xff0c 在操作系统有两个空间即user space和kernal space 每个Tcp socket连接在内核中都有一个发送缓冲区和接收缓冲区 xff0
  • QtCreator修改项目构建目录

    使用QtCreator编译Qt项目时 xff0c 如有需求修改编译过程文件 xff08 即Makefile o exe等文件 xff09 存放目录 xff0c 简单在工具 gt 选项 gt 构建和运行中修改Default build dir
  • QT中删除信号于槽的连接

    如果是在UI里建立的 xff0c 那就在下面这个函数里删除连接槽函数的对应行 void MainForm qt static metacall QObject o QMetaObject Call c int id void a if c
  • C++位操作中按位置0、置1、取反操作

    一 指定的某一位数置1 宏 define setbit x y x 61 1 lt lt y 二 指定的某一位数置0 宏 define clrbit x y x amp 61 1 lt lt y 三 指定的某一位数取反 宏 define r
  • LED恒流驱动IC汇总

    这几天在找LED恒流驱动芯片 xff0c 无意间在LED网论坛上发现这个帖子 xff0c 分享给大家 xff01 LED恒流IC芯片大盘点 韩国LDT LD1016 16位最大90mA LED屏幕 护栏灯管恒流驱动IC LD1048 48位
  • qt C++中指针自动释放内存及程序中的内存操作、管理

    程序加载到内存后代码存储到代码区 xff0c 并将全局变量 静态变量初始化到全局 静态内存区 xff0c 然后会分配2M左右的栈内存区用于存储局部变量 xff0c 并在运行时根据需要可以在堆内存区 空闲内存区及硬盘的虚拟内存区 申请空间 程
  • sqlite数据库文件提示损坏修复方法

    第1章 说明 1 1 下载SQLite Tools 1 2 运行 2 注意 xff1a 为了方便 xff0c 可把要修复的数据库文件直接放到sqlite3 exe路径下然后运行sqlite3 exe就不用输入具体路径了 sqlite3 ex
  • 常用的dos网络命令总结

    一 ping 主要是测试本机TCP IP协议配置正确性与当前网络现状 ping命令的基本使用格式是 xff1a ping IP地址 主机名 域名 t a n count l size t xff1a 连续对IP地址 主机名 域名执行Ping
  • C#中隐藏窗体并执行窗体逻辑的方法

    c 隐藏窗体方方法 this WindowState 61 FormWindowState Minimized this ShowInTaskbar 61 false base SetVisibleCore true 示例如下 Task F
  • qt中xe运行缺少组件,Qt-c++桌面编程报错:qt.qpa.plugin: Could not find the Qt platform plugin “windows“ in ““,最终解决方案

    报以上错误是因为编译出来的exe程序缺少一些qt的组件 xff0c 需要补全放到exe同级目录下即可 编译库 xff1a Qt GUI xff0c qt5 12 1 软件类型 xff1a Qt application xff0c qt桌面软
  • C#中异步窗体的调用的几种方法

    在子线程里创建的窗体在其他线程里调用的异步调用操作方法 xff1a 1 在另一个线程里找到异步窗体句柄 xff0c 向其传送自定义消息ID 异步窗体的消息循环需要重写加入相应自定义方法 2 找到异步线程里的窗体 xff0c 用该窗体类型直接
  • C++中UDP通讯详解

    C 43 43 Socket编程及TCP UDP通信代码实现 一 简介 Socket编程的目的是使网络的进程进行通信 xff0c 基于TCP IP协议簇 xff0c 通过三元组 xff08 ip地址 协议 端口 xff09 标志进程 xff
  • UDP中connect()和send()函数详解

    UDP也是有connect方法的 xff0c 下面为大家讲解一下这个connect方法 我们知道UDP是无连接的 xff0c 它可以给多个IP发送数据包 xff0c 包括广播地址或者多播通信的实现 xff0c 而这些是TCP IP无法实现的
  • UDP传输报文大小详解

    UDP的传输方式 xff1a 面向报文 面向报文的传输方式决定了UDP的数据发送方式是一份一份的 xff0c 也就是应用层交给UDP多长的报文 xff0c UDP就照样发送 xff0c 即一次发送一个报文 那么UDP的报文大小由哪些影响因素
  • UDP通信概念问题

    1 UDP 多个线程 通过同一个端口接收和发送数据 xff0c 是双向不收影响的还是在时序上错开的 xff1f 如题 xff0c 比如接收到了不同地址的的UDP数据 xff0c 现在通过多个线程从同一个端口回发数据 那么数据的收发都是同一个
  • KMeans算法-手写数字图像识别

    数据聚类是无监督学习的主流应用 最经典并易用的聚类模型 xff0c 是K means算法 该算法要求我们预设聚类的个数 xff0c 然后不断更新聚类中心 xff1b 经过几轮迭代后 xff0c 让所有数据点到其所属聚类中心距离的平方和趋于稳
  • C++中创建二维数组的几种方法

    一 用new申请内存空间 int dp 61 new int n 动态申请二维数组nxm for int i 61 0 i lt n 43 43 i dp i 61 new int m 二 用malloc申请内存空间 1 int len 6

随机推荐