一般来说,我们发出信号使用emit这个关键字来操作,但是会发现,emit并不算一个调用,所以它没有返回值。那么如果我们发出这个信号想获取一个返回值怎么办呢?
两个办法:1.通过出参形式返回,引用或者指针的方式带回;比如emit sig(int& i)或者emit sig(void* pointer),但是这个方法有一个弊端,稍后介绍第二种方式会提醒。
2.通过qt自带的invoke机制调用:参考文档对QMetaObject::invokeMethod的说明:Invokes the member (a signal or a slot name) on the object obj.也就是说回调是可以回调信号或者槽的。一般来说,我们使用invokeMethod是在子线程需要调度UI操作的时候,因为UI操作只能在主线程中使用(否则会出现未定义错误)。大部分情况下,我们把UI操作封装在一个槽里,用回调方式来调度。同样信号也可以用这种方式,但是有几点需要注意的是,1.调用回调的连接方式:如果信号和连接槽在一个线程内,那么必须用Qt::DirectConnection或者Qt::AutoConnection,这样的话,保证信号回调后,线程会等待信号连接槽执行完毕,才可能取到我们需要的返回值;如果使用了Qt::QueuedConnection,那么信号只是负责把事件交给事件队列,然后马上做出返回,这样,是否有返回值就无法确定了(这也就是第一个方法的弊端).如果信号和槽在两个线程中,那么首先肯定不能使用Qt::DirectConnection,除非你很清楚连接槽的动作是否保证了线程安全。但根据第一条的说明,也不能使用Qt::QueuedConnection。不过还好qt提供了一个额外的连接方式就是Qt::BlockingQueuedConnection,这个连接方式会阻塞住发射信号的线程一直等到队列连接槽返回后,才会恢复阻塞,这样就可以保证我们能拿到真正的返回值。(但是使用这种方式需要你清楚的知道,发射线程是否允许阻塞和连接槽是否对这个阻塞线程有什么特别的操作,一般来说,如果这个线程并不是由你自己控制的话,不要随便尝试去阻塞别人的线程,因为你并不清楚别人线程的执行逻辑)
调用方式大致代码如下bool bReturn; QMetaObject::invokeMethod(&object, "sig", Qt::DirectConnection/*Qt::QueuedConnection*/, Q_RETURN_ARG(bool, bReturn), Q_ARG(int, i));
/************************
QT 信号槽返回值
一、方法
获取信号槽返回值的方法有三种:
(1)通过形参的方式,传递引用或者指针。(未写例子)
(2)直接在信号发出位置获取返回值。(下面有例子)
(3)通过QMetaObject::invokeMethod方法。(具体使用可以参考QT的帮助文档或者看下面的代码实例)
二、代码
#ifndef WRITETHREAD_H
#define WRITETHREAD_H
#include <QObject>
class WriteThread : public QObject
{
Q_OBJECT
public:
WriteThread(QObject *parent = nullptr);
public slots:
QString slotWrite(QString str);
signals:
};
#endif // WRITETHREAD_H
#include "writethread.h"
#include <QDebug>
#include <QDateTime>
#include <QThread>
WriteThread::WriteThread(QObject *parent) : QObject(parent)
{
}
QString WriteThread::slotWrite(QString str)
{
qDebug()<<"执行了slotWrite 子线程id:"<<QThread::currentThreadId();
QThread::sleep(3);
QString date = "函数返回时间:"+QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss");
return str + "\n" + date + "\n";
}
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>
#include "writethread.h"
#include <QThread>
QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACE
class MainWindow : public QMainWindow
{
Q_OBJECT
public:
MainWindow(QWidget *parent = nullptr);
~MainWindow();
signals:
QString signalWrite(QString str);
private slots:
void on_pushButton_clicked();
void on_pushButton_2_clicked();
private:
Ui::MainWindow *ui;
QThread *m_pThread;
WriteThread *m_pWriteThread;
};
#endif // MAINWINDOW_H
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include<QDebug>
#include<QDateTime>
MainWindow::MainWindow(QWidget *parent)
: QMainWindow(parent)
, ui(new Ui::MainWindow)
{
ui->setupUi(this);
qDebug()<<"主线程id:"<<QThread::currentThreadId();
m_pThread = new QThread;
m_pWriteThread = new WriteThread;
m_pWriteThread->moveToThread(m_pThread);
connect(this,&MainWindow::signalWrite,m_pWriteThread,&WriteThread::slotWrite,Qt::BlockingQueuedConnection);
m_pThread->start();
}
MainWindow::~MainWindow()
{
delete ui;
m_pThread->quit();
m_pThread->wait();
delete m_pThread;
m_pThread = NULL;
}
//写值 方法1:直接获取返回值
void MainWindow::on_pushButton_clicked()
{
qDebug()<<"点击了写值按钮";
QString str = "信号发出时间:"+ QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss");
qDebug()<<"信号发出了";
QString ret = signalWrite(str);
qDebug()<<"返回值为:"<<ret;
ui->textEdit->append(ret);
}
//写值 方法2:QMetaObject::invokeMethod
void MainWindow::on_pushButton_2_clicked()
{
qDebug()<<"点击了写值按钮";
QString str = "信号发出时间:"+ QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss");
QString ret;
QMetaObject::invokeMethod(m_pWriteThread, "slotWrite", Qt::BlockingQueuedConnection,
Q_RETURN_ARG(QString, ret),
Q_ARG(QString, str));
qDebug()<<"返回值为:"<<ret;
ui->textEdit->append(ret);
}
ui界面
结果
三、补充
①信号发送者和接受者在同一个线程,也就是直接连接,参数为Qt::DirectConnection。此时为同步调用,信号发送之后会等待槽函数的执行,可接收到返回值。
②当信号发送者和接受者不在同一个线程,也就是队列连接,参数为Qt::QueuedConnection时。此时为异步调用,信号发出之后,把事件交给事件队列,不等待槽函数的执行直接返回,不能接收到返回值。
③当信号的发送者和接受者不在同一个线程时,可以使用Qt::BlockingQueuedConnection参数,该参数会阻塞信号发送者所在线程,直到槽函数执行完成之后才会往下执行,可以接收到返回值。需要注意:如果信号的发送者和接受者在一个线程,使用该参数会造成程序死锁。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)