QThread 线程间通信:连接到 &QThread::quit 与连接到 lambda [&thread] thread->quit(); 的奇怪行为

Posted

技术标签:

【中文标题】QThread 线程间通信:连接到 &QThread::quit 与连接到 lambda [&thread] thread->quit(); 的奇怪行为【英文标题】:QThread interthread communication: strange behavior connecting to &QThread::quit vs connecting to a lambda [&thread] thread->quit();QThread 线程间通信:连接到 &QThread::quit 与连接到 lambda [&thread] thread->quit(); 的奇怪行为 【发布时间】:2016-05-28 12:34:09 【问题描述】:

我有以下设置:

int main(int argc, char *argv[])

    QCoreApplication a(argc, argv);

    // Create the DBManager that will live for the entire
    // duration of the application
    auto dbManagerThread = std::make_unique<QThread>();
    auto dbManager = std::make_unique<DBManager>();
    dbManager->moveToThread(dbManagerThread.get());
    dbManagerThread->start();


    // for the initialization of the application, create the
    // InitHelper object that will utilize the DBManager
    auto initHelper = new InitHelper();
    auto initHelperThread = new QThread();
    initHelper -> moveToThread(initHelperThread);
    // wire InitHelper and its thread
    QObject::connect(initHelperThread, &QThread::started, initHelper, &InitHelper::run);
    QObject::connect(initHelper, &InitHelper::finished, [&initHelperThread]  initHelperThread->quit(););
    QObject::connect(initHelper, &InitHelper::finished, initHelper, &InitHelper::deleteLater);
    QObject::connect(initHelperThread, &QThread::finished, initHelperThread, &QThread::deleteLater);

    // wire InitHelper and DBManager
    QObject::connect(initHelper, &InitHelper::queryDB, dbManager.get(), &DBManager::processQuery);
    QObject::connect(dbManager.get(), &DBManager::queryResult, initHelper, &InitHelper::processQueryResult);


    // block until all information is gathered
    initHelperThread->start();
    initHelperThread->wait();
    std::cout << "Initialization completed." << std::endl;

    // cleanup
    dbManagerThread->quit();
    QObject::connect(dbManagerThread.get(), &QThread::finished, &a, &QCoreApplication::quit);
    return a.exec();

我的想法是我有 DBManager,它正在执行异步数据库访问,因此它在整个应用程序中都被使用。在应用程序初始化期间,我需要从数据库中检索一些信息,并且我想使用已经存在的 DBManager。

但是,由于我需要数据库中的信息来创建所有其他对象,我想阻止主线程的进一步执行,直到我的 InitHelper 从 DBManager 检索到所有必需的信息。

由于上面的代码完全符合预期,我认为 InitHelper 和 DBManager 的实现方式并不重要,我在这里省略了它们。

令我感到困惑的是,我需要在“wire InitHelper 及其线程”部分的第二行中使用 lambda。如果我替换

QObject::connect(initHelper, &InitHelper::finished, [&initHelperThread] initHelperThread->quit(););

通过

QObject::connect(initHelper, &InitHelper::finished, initHelperThread, &QThread::quit);

显然线程永远不会关闭,因此“初始化完成”永远不会打印到标准输出。

这是使用 lambda 连接时的输出:

InitHelper: Initialization started...
DBManager: processing query...
InitHelper: processing query result
Initialization completed.

而直接连接到插槽时,我得到以下输出:

InitHelper: Initialization started...
DBManager: processing query...
InitHelper: processing query result

有人知道为什么连接到 lambda 而不是直接连接到插槽时会有区别吗?以及如何在没有 lambda 的情况下编写连接?

【问题讨论】:

【参考方案1】:

在您的代码中:

QObject::connect(initHelper, &InitHelper::finished, [&initHelperThread]  initHelperThread->quit(););

类似于:

QObject::connect(initHelper, &InitHelper::finished, initHelperThread, &QThread::quit, Qt::DirectConnection);

这基本上意味着你是直接调用函数而不是使用事件循环。

我认为,您的“正确”代码不起作用(即,当您使用 Qt::QueuedConnection - 这是默认值时)的原因是因为您没有通过调用 a.exec(); 启动 Qt 事件循环运行/p>

为了解决这个问题,我将删除您的 wait() 函数并将您的“启动”代码移动到启动类(有点像您的 initHelper)。将 initHelper 完成后的信号连接到启动类的 start() 函数(或任何你想调用的函数)。

编辑

您也许可以这样做(只需阅读类似的内容):

int ret = a.exec();
initHelperThread->wait();
std::cout << "Initialization completed." << std::endl;
    :
  do other stuff
    :
return ret;

【讨论】:

你是对的,将 Qt::DirectConnection 添加到时隙连接调用中就可以了。 我添加了一个编辑,我看到了类似的东西,对你来说可能更快:) 我认为只添加 Qt::DirectConnect 是最简单的方法。在 wait() 之前运行 a.exec() 不起作用。 真的吗? - 稍后我将不得不玩弄它。所以,通常我会建议不惜一切代价避免跨线程的直接连接 - 糟糕的习惯......但因为这是一次运行并且只有在启动时我们才能忍受它。但仅供参考,请不要习惯这样做,它会给你带来很多麻烦:p(以防你不知道) QCoreApplication 确实需要一个“初始化完成,事件循环正在运行”信号,也许是通过将特定事件放入队列中。一旦循环运行,它将发出单次信号。唉...

以上是关于QThread 线程间通信:连接到 &QThread::quit 与连接到 lambda [&thread] thread->quit(); 的奇怪行为的主要内容,如果未能解决你的问题,请参考以下文章

Qt信号发射和QThread执行流程

Qt例子,线程间通信,如何在线程外部对线程进行控制,问题请看问题补充,多谢了先

QT多线程及通过事件进行通信(通过自定义事件,然后QApplication::postEvent给主界面,我之前用的是信号槽)

在 QThread::run 中间调用了一个插槽

PyQt - 主要挂在 QThread

完成后将 QRunnable 连接到函数/方法