QT信号槽连接之不同线程之间的信号槽连接方式
QT中信号槽的连接方式有五种:Qt::AutoConnection信号槽默认的连接方式,如果信号与槽在同一线程,就自动采用Qt::DirectConnection,如果信号与槽不在同一线程,将自动采用Qt::QueuedConnection的连接方式。Qt::DirectConnectionQt::DirectConnection表示一旦信号产生,立即执行槽函数。 (如果主线程发送信号给...
QT中信号槽的连接方式有五种:
Qt::AutoConnection
信号槽默认的连接方式,如果信号与槽在同一线程,就自动采用Qt::DirectConnection,
如果信号与槽不在同一线程,将自动采用Qt::QueuedConnection的连接方式。
Qt::DirectConnection
Qt::DirectConnection表示一旦信号产生,立即执行槽函数。 (如果主线程发送信号给子线程,则这个参数要使用 Qt::DirectConnection ,槽函数在子线程中立即执行)
Qt::QueuedConnection
在不同的线程中处理, Qt::QueuedConnection表示将发送信号给接受线程,并且进入接收线程的队列,等候处理,像是Win32中PostMessage
Qt::BlockingQueuedConnection
在不同的线程中处理,当前线程信号发送后,会阻塞等待,接收线程处理完成后才会返回,发送线程会进行下一步的处理 。
但是如果在一个线程中使用这种信号槽的连接方式,会是怎么样的呢?
在运行的时候,程序会一直卡在发送信号的地方,并不会报错但是,会输出一下信息:
Qt: Dead lock detected while activating a BlockingQueuedConnection: Sender is QtManager(0xfc3f8c3a40), receiver is QtManager(0xfc3f8c3a40)
出现了死锁现象。
信号槽的连接可以在在一个地方,但是发送信号的地方和槽函数的地方必须在两个线程中才可以
Qt::UniqueConnection
Qt::UniqueConnection表示只有它不是一个重复连接,连接才会成功。如果之前已经有了一个链接(相同的信号连接到同一对象的同一个槽上),那么连接将会失败并将返回false。
例如:
#pragma once
#include <QThread>
class MyThread : public QThread {
Q_OBJECT
public:
MyThread(QObject *parent);
~MyThread();
void run()override;
private:
int _count = 0;
};
#include "MyThread.h"
#include "QtManager.h"
#include <QDebug>
MyThread::MyThread(QObject *parent)
: QThread(parent) {
}
MyThread::~MyThread() {
}
void MyThread::run() {
while (true) {
msleep(800);
_count++;
Mgr()->setTextInThread(QString::number(_count));
qDebug() << QString("MyThread:%1").arg(_count);
}
}
#pragma once
#include <QString>
class IText {
public:
IText();
~IText();
virtual void setText(QString t);
};
#include "IText.h"
IText::IText() {
}
IText::~IText() {
}
void IText::setText(QString t) {
}
#pragma once
#include <QtWidgets/QMainWindow>
#include "ui_QtGuiThread.h"
#include "IText.h"
class MyThread;
class QtGuiThread : public QMainWindow ,public IText {
Q_OBJECT
public:
QtGuiThread(QWidget *parent = Q_NULLPTR);
virtual void setText(QString t)override;
private:
void init();
private slots:
void slotStart();
private:
Ui::QtGuiThreadClass ui;
MyThread* _myThread = nullptr;
};
#include "QtGuiThread.h"
#include "MyThread.h"
#include "QtManager.h"
QtGuiThread::QtGuiThread(QWidget *parent)
: QMainWindow(parent) {
ui.setupUi(this);
init();
}
void QtGuiThread::setText(QString t) {
ui.lineEdit->setText(t);
}
void QtGuiThread::init() {
Mgr()->setIText(this);
connect(ui.pushButton, SIGNAL(clicked()), this, SLOT(slotStart()));
_myThread = new MyThread(nullptr);
}
void QtGuiThread::slotStart() {
_myThread->start();
}
aaa
开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!
更多推荐
所有评论(0)