当前位置: 首页 > news >正文

Qt多线程编程实践总结:QtConcurrent与moveToThread应用场景对比(可以说都是干货)

在使用Qt进行多线程开发时,常见的两种方式是 QtConcurrent::run 和 QObject::moveToThread。虽然它们都能实现“多线程”,但其底层机制、使用场景以及适用对象却大相径庭。本文结合实际开发经验,对 Qt 多线程机制的关键点进行总结,帮助开发者更合理地选择和使用这两种方式。代码详见https://gitee.com/zhang_jie_sc/LGHWTest

1.QtConcurrent::run:适合小任务、无状态操作

特点:

  • 使用线程池 (QThreadPool) 自动管理线程;
  • 每次调用 QtConcurrent::run(…),可能由不同线程执行;
  • 不适合涉及成员变量或状态的对象方法;
  • 非事件驱动,无法处理 Qt 信号槽、定时器等事件。

示例:

	QtConcurrent::run(worker, &ImageWriterWorker::writeFrame, frame);

注意:
每次调用都可能在不同线程执行。

2.moveToThread + invokeMethod:适合长期运行的后台对象

特点:

  • 将一个 QObject 派发到专门线程中;
  • 使用事件循环异步处理信号和调用;
  • 可处理 Qt 信号槽、事件、定时器等;
  • 调用该对象的方法时,可通过 invokeMethod 安全切换至目标线程。

示例:

worker->moveToThread(thread);
QMetaObject::invokeMethod(worker, "writeFrame",Qt::QueuedConnection,Q_ARG(FrameData, frame));

所有调用都在 worker 所在线程中有序执行,适用于后台日志写入、图像处理、数据采集等任务。

3.线程归属

QObject 属于哪个线程,由创建它的线程决定;调用它的方法,则取决于调用方式。在C#的winform和wpf也是这样的,控件只能由创建它的线程进行操作,否则都是都过委托实现。

3.1. 对象的“线程归属”(线程亲缘性)

在 Qt 中,每个 QObject 都属于某一个线程 —— 即“线程亲缘性”(thread affinity):
- 默认情况下,QObject 在哪个线程中创建,就属于哪个线程;
- 可以通过 obj->moveToThread(targetThread) 显式更改其归属;
- 属于某个线程的对象,其槽函数在该线程的事件循环中执行,前提是使用了 QueuedConnection 或跨线程信号。
- 线程归属决定了事件处理的线程。

3.2. 方法调用行为

  • 同线程直接调用: 立即在当前线程调用;
  • 跨线程(Queued Connection): 方法调用变成一个事件,排入目标线程的事件队列;
  • Auto Connection(默认): 若在同线程中,则为直接调用;否则自动变为队列调用。

如何从主线程调用后端对象的方法?

QObject::connect(sender, SIGNAL(...), receiver, SLOT(...), Qt::QueuedConnection);

或者:

QMetaObject::invokeMethod(obj, "someSlot", Qt::QueuedConnection, ...);

这样即使从主线程调用,也能保证目标函数在后台线程中执行,直接在主线程调用,会使程序出现不稳定状态。

3.3. 常见误区

错误做法说明
在主线程创建对象,moveToThread 后仍直接调用方法仍然是在主线程执行代码,违背线程封装
不设置事件循环就 moveToThread无法处理信号、队列事件,线程形同虚设

QThread::start()默认会在线程中启动一个事件循环,正常调用就行了。
在这里插入图片描述

4.使用建议

场景推荐方式原因
小量、一次性后台任务QtConcurrent::run简洁高效,适合无状态计算
长时间后台服务对象moveToThread+invokeMethod保证线程稳定性
对象需接收信号/定时器moveToThread线程需拥有事件循环
每次调用都能独立执行QtConcurrent::run线程池调度,性能佳
多线程操作有状态对象避免 QtConcurrent存在线程安全问题

5.直接继承QThread为什么不被推荐

5.1.误解线程的角色

  • QThread 的职责是管理线程的生命周期(创建、启动、停止),而不是直接承载业务逻辑。
  • 很多人在继承 QThread 后,把业务逻辑写在 run() 里,这样线程内就不能再拥有事件循环(除非自己加 exec())。

5.2.没有事件循环,无法响应 signal-slot、定时器等机制

5.3.线程中的对象容易绑定到错误线程(还没验证,有兴趣可以验证一下)

如果你在 QThread 子类的 run() 中创建对象,这些对象自动归属于创建它们的线程(主线程),除非手动调用 moveToThread(),容易造成线程混乱或警告:QObject::startTimer: timers cannot be started from another thread。

在这里插入图片描述

相关文章:

  • 接口(API)开发核心知识点
  • Qt —— 使用Enigma Virtual Box将Qt程序打包为独立可运行exe(附:完整打包方法且完美运行)
  • CSRF防范歪招
  • 分区器介绍
  • Java集合框架详解与使用场景示例
  • PyInstaller 打包后 Excel 转 CSV 报错解决方案:“excel file format cannot be determined“
  • uniapp(vue3)动态计算swiper高度封装自定义hook
  • Foupk3systemX5OS TXW8移动设备
  • UE5中制作动态数字Decal
  • While语句数数字
  • 互信息:揭秘变量间的隐藏关联
  • 5.13本日总结
  • windows 强行终止进程,根据端口号
  • 【Linux 系统调试】系统的动态跟踪工具--SystemTap
  • 系统平衡与企业挑战
  • C++ 字符格式化输出
  • Linux 系统安全基线检查:入侵防范测试标准与漏洞修复方法
  • 【递归、搜索与回溯】专题一:递归(二)
  • SparkSQL 连接 MySQL 并添加新数据:实战指南
  • 微服务八股(自用)
  • 佩斯科夫:若普京认为必要,将公布土耳其谈判俄方代表人选
  • 通化市委书记孙简升任吉林省副省长
  • 【社论】个人破产探索,要守住“诚实而不幸”的底线
  • 沈阳卫健委通报“健康证”办理乱象:涉事医院已被立案查处
  • 新闻1+1丨婚姻登记服务,如何跑出幸福加速度?
  • 郑州通报涉“健康证”办理有关问题查处进展情况