QT中connect连接信号槽中使用lambda函数注意点

注意点:

信号槽的的默认连接方式为autoconnect的方式.

当信号的发射位置(也即emit写的位置)与槽函数的接收者在同一个线程中时,默认就是直连的连接模式, 当发送信号时,立即执行槽函数.

当信号的发射位置(也即emit写的位置)与槽函数的接收者在不同的线程中时,默认就是队列的连接模式, 当发送信号时,将把信号发送到接收者线程的事件队列中,等事件队列执行到该信号时再执行槽函数.

重点:不是信号发送者与槽函数的接收者所处线程来判断,而是信号发送的位置与接收者所处线程来判断.

cpp 复制代码
  connect(m_pGenCode, &QThread::started, 
      this, &qwaGrindProcessDS::slotStartGenCode);

  connect(m_pGenCode, &QThread::started,
      [this]() {qInfo() << "started lambda:" << QThread::currentThreadId(); });
  connect(m_pGenCode, &QThread::started,this,
      [this]() {qInfo() << "started lambda:" << QThread::currentThreadId(); });

m_pGenCode通过继承QThread的方式定义的QThread子类对象指针, 在主窗口中定义.通过重写run函数,来实现的多线程的方式.

所以m_pGenCode与this是在一个线程. 如果需要在不同的线程中,则需要采用movetothread的方式创建线程.2种方式各有优缺点.

cpp 复制代码
connect(m_pGenCode, &QThread::started, this, &qwaGrindProcessDS::slotStartGenCode);

QThread::started()信号是在调用了m_pGenCode->start()时发送的,当start()执行时,会创建一个新的线程,同时发送一个started()信号.所以QThread::started()信号是在新的线程中发送的.

this在旧的主线程中, 所以默认是采用队列的连接模式.

cpp 复制代码
connect(m_pGenCode, &QThread::started,[this]() {
qInfo() << "started lambda:" <<QThread::currentThreadId(); 
});

直接在信号后面写lambda函数,这时候没有接收者,默认这个lambda函数将在新的线程中执行,也即发射信号的线程中执行.采用直接连接模式.

cpp 复制代码
  connect(m_pGenCode, &QThread::started,this,
      [this]() {qInfo() << "started lambda:" << QThread::currentThreadId();})

在接收者位置多了this,即主窗口线程对象. 这时候发送者与接收者虽然都在同一个线程中,但是信号的发送位置是在新线程中,所以默认为队列连接模式. 这个lambda函数将在this所在线程中执行.

相关推荐
卷无止境1 天前
C++ 的Eigen 库全解析
c++
卷无止境1 天前
现代 C++特性大盘点:一门脱胎换骨的老语言
c++·后端
郝学胜_神的一滴1 天前
CMake 27:缓存变量的特性、语法、类型与实操全解
c++·cmake
Quz2 天前
QML Hello World 入门示例
qt
博客18003 天前
酷宝的使用方法,超好用的免费界面库,C++、MFC可用
c++·mfc·界面库·库来帮·酷宝
郝学胜_神的一滴3 天前
CMake 026:属性体系精讲、四大作用域全解 & 实战代码落地
c++·cmake
众少成多积小致巨4 天前
JNI (Java Native Interface) 技术手册中文参考指南
android·java·c++
xcyxiner5 天前
DicomViewer (dcmtk读取dcm文件)5
qt
xcyxiner6 天前
DicomViewer (后台线程处理文件)4
qt
xcyxiner6 天前
DicomViewer (添加模型类)3
qt