Qt 多线程误区:QThread 任务未正确执行及收尾回调未触发的原因分析

问题背景

在开发 Qt 项目时,我们希望并行执行多个任务来提高效率,并在所有任务完成后进行收尾处理。为此,我们使用 QThread 来执行任务,并在 QThread::finished 事件触发时调用收尾函数。

原始代码

void MyClass::runTaskInThread(void (MyClass::*task)(), void (MyClass::*onFinished)())
{
    QThread *thread = new QThread(this);
    connect(thread, &QThread::started, this, task);  // ❌ 错误:task 仍然在主线程执行!
    connect(thread, &QThread::finished, this, [=]() {
        (this->*onFinished)(); // 触发完成回调
        thread->deleteLater(); // 自动清理线程
    });
    thread->start();
}

void MyClass::startTask()
{
    runTaskInThread(&MyClass::task1, &MyClass::task1Finished);
    runTaskInThread(&MyClass::task2, &MyClass::task2Finished);
    runTaskInThread(&MyClass::task3, &MyClass::task3Finished);
}

void MyClass::task1() {}
void MyClass::task2() {}
void MyClass::task3() {}

void MyClass::taskFinishedHandle()
{
    if(task1Finished && task2Finished && task3Finished)
        doSomething();
}

void MyClass::task1Finished() { task1Finished = true; taskFinishedHandle(); }
void MyClass::task2Finished() { task2Finished = true; taskFinishedHandle(); }
void MyClass::task3Finished() { task3Finished = true; taskFinishedHandle(); }

发现的问题

在实际运行中,task1Finishedtask2Finishedtask3Finished从未被执行,导致 taskFinishedHandle 也不会触发。

错误分析:QThread 默认不会运行任务

在 Qt 中,QThread默认不会自动运行任何代码,它只是提供了一个独立的事件循环。

原始代码的关键错误:

connect(thread, &QThread::started, this, task);  // ❌ 错误

这个 connect 绑定的 task 仍然是 this(主线程)的方法,因此 task()仍然在主线程执行,而 QThread 只是空跑了一下,没有执行任何任务。

为什么 QThread 变成了“空壳线程”?

  1. thread->start(); 只是启动了 QThread,进入 事件循环,但 不会自动运行任务
  2. connect(thread, &QThread::started, this, task); 触发 task(),但 this 仍然是 主线程,所以 task()仍然在主线程里执行
  3. task() 运行完后,线程 实际上并没有执行任何代码,于是 QThread 认为工作完成,触发 finished,然后销毁。
总结:你只是新建了一个 QThread,但任务仍然在主线程执行,导致 QThread 变成了一个“空壳”。

正确的实现方式

方法 1:使用 moveToThread 让任务运行在 QThread

void MyClass::runTaskInThread(void (MyClass::*task)(), void (MyClass::*onFinished)())
{
    QThread *thread = new QThread(this);
    QObject *worker = new QObject();  // 创建一个独立的 worker
    worker->moveToThread(thread); // 让 worker 运行在新线程

    connect(thread, &QThread::started, worker, [=]() {
        (this->*task)();  // 现在 task() 会在新线程中执行
        QMetaObject::invokeMethod(this, onFinished, Qt::QueuedConnection);  // 让 onFinished 在主线程运行
        thread->quit();
    });

    connect(thread, &QThread::finished, thread, &QObject::deleteLater);
    thread->start();
}

方法 2:使用 QtConcurrent::run

#include <QtConcurrent>

void MyClass::runTaskInThread(void (MyClass::*task)(), void (MyClass::*onFinished)())
{
    QtConcurrent::run([=]() {
        (this->*task)();   // 任务会在新线程中执行
        QMetaObject::invokeMethod(this, onFinished, Qt::QueuedConnection);  // 让回调在主线程运行
    });
}

两种方法的区别

方法

适用场景

优势

moveToThread

需要更灵活的线程管理(如持续运行任务)

线程控制权更强,可绑定 QObject 事件循环

QtConcurrent::run

任务是一次性的,无需额外管理线程

代码更简洁,自动管理线程,避免 QThread 内存泄漏

总结

错误原因:

  • 直接 connect(thread, &QThread::started, this, task); 任务仍然在 主线程 运行,导致 QThread没有执行任何代码

正确做法:

  1. moveToThread 确保任务在 QThread 里运行。
  2. 更推荐用 QtConcurrent::run,代码简洁,自动管理线程,不会导致 QThread 泄漏。

如果你的 task1task2task3 仍然卡在主线程,请一定要改用 QtConcurrent::runmoveToThread

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

丰年稻香

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值