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

【Qt】QThread总结

目录

    • 成员函数
    • 创建方式
      • 方式一
      • 方式二
      • 方式三
      • 注意
    • example
    • 总结
    • 参考文章

成员函数

创建方式

方式一

QThread 静态成员create

auto thd = QThread::create([]{});

方式二

继承QThread类,重写run

  1. run函数它作为线程的入口,也就是线程从run()开始执行,我们打算在线程中完成的工作都要写在run()函数中,个人认为可以把run()函数理解为线程函数。这也就是子类覆写基类的虚函数,基类QThread的run()函数只是简单启动exec()消息循环。
  2. Qthread中其他函数均不是在子线程中执行

方式三

movemoveToThread使用信号与槽方式来实现多线程

class Work : public QObject
{
public slots:
    void doWorking()
    {
        for (size_t i = 0; i < 10; i++)
        {
            std::cout << i << ", thd ID=" << QThread::currentThread() << std::endl;
        }

        std::cout << "-----------------------------" << std::endl;
    }

    void doOtherWorking()
    {
        for (size_t i = 0; i < 10; i++)
        {
            std::cout << i << ", thd ID=" << QThread::currentThread() << std::endl;
        }

        std::cout << "doOtherWorking -----------------------------" << std::endl;
    }
};

auto pThd = new QThread;
Work work;
work.moveToThread(pThd);
QObject::connect(&startButton, &QPushButton::clicked, [&]
                 { pThd->start(); });

QObject::connect(&thdButton, &QPushButton::clicked, &work, &Work::doWorking);
QObject::connect(&thdButton, &QPushButton::clicked, &work, &Work::doOtherWorking);
    
std::cout << QThread::currentThread() << std::endl;

注意

  1. Work实例化时不可以指定父对象
  2. 当一个变量需要在多个线程间进行访问时,最好加上voliate关键字,以免读取到的是旧的值。当然,Qt中提供了线程同步的支持,比如互斥锁之类的玩意,使用这些方式来访问变量会更加安全
  3. 一定要用信号槽机制,别想着直接调用,你会发现并没有在子线程中执行

example

  1. 非阻塞式线程运行
QEventLoop loop;
QtConcurrent::run([&]()
{
    doSomething();
    loop.quit();
});
loop.exec();

上述写法存在两个问题

  1. QtConcurrent::run()执行后线程立即开始执行,当doSomething()很快时,loop.quit()的执行时间可能早于loop.exec(),这样事件循环就再也不会退出了
  2. 当doSomething因为某些原因crash时,如果没有导致程序崩溃,loop.quit无法被执行,事件循环同样无法退出

下面给出优化后的代码

QEventLoop loop;
QThread* pThread = QThread::create([&]()
{
	doSomething();
});
connect(pThread, &QThread::finished, &loop, &QEventLoop::quit);
pThread->start(); // 线程开始执行
loop.exec();

总结

  1. 第一种方法适合创建临时任务,用到了lambda表达式
  2. 第二种方法适合在线程中处理单一事件,其逻辑简单(只需要新建一个继承自QThread类的对象,重写run函数,然后启动即可)
  3. 第二种方法,我们可以自定义带参的子线程运行函数,代码简洁;随意修改需要在哪个线程中运行,代码灵活

参考文章

QT从入门到入土(五(1))——多线程(QThread) - 唯有自己强大 - 博客园


http://www.kler.cn/a/505751.html

相关文章:

  • Spring Boot教程之五十五:Spring Boot Kafka 消费者示例
  • STM32 FreeRTOS时间片调度---FreeRTOS任务相关API函数---FreeRTOS时间管理
  • 49_Lua调试
  • 【声音场景分类--论文阅读】
  • w160社区智慧养老监护管理平台设计与实现
  • Spring Boot 应用开发入门
  • flutter R库对图片资源进行自动管理
  • c#删除文件和目录到回收站
  • 【Linux系统编程】—— 自动化构建工具Makefile指南
  • rtthread学习笔记系列(3) -- FINSH模块
  • 寄存器 reg
  • 【学习笔记】GitLab 使用技巧和说明和配置和使用方法
  • [操作系统] 深入理解约翰·冯·诺伊曼体系
  • DNS介绍(1):基本概念
  • 如何确保API调用安全
  • Flink (三):核心概念(并行度、算子链、任务槽)
  • 算法面试准备 - 手撕系列第一期 - Softmax
  • WPF-01理解XAML
  • 不用PLC和板卡,一台电脑就可以控制伺服
  • Vue.js 动态组件与异步组件
  • 字典和 JSON 文本的格式区别
  • 【漫话机器学习系列】044.热点对特性的影响(Effect Of One Hot On Feature Importance)
  • Rust 正则表达式完全指南
  • zerox - 使用视觉模型将 PDF 转换为 Markdown
  • 机器学习中的凸函数和梯度下降法
  • 海康MV-EB435i立体相机SDK安装(ROS 2)