Qt实现多线程 QThread

2024-02-26 18:52
文章标签 实现 qt 多线程 qthread

本文主要是介绍Qt实现多线程 QThread,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

    QThread类提供了一种独立于平台的方式来管理线程。QThread对象管理程序中的一个控制线程。多线程在run()中开始执行。默认情况下,run()通过调用exec()启动事件循环,并在线程内运行Qt事件循环。可以通过使用QObject::moveToThread()将工作对象移动到线程中来使用它们。

Qt创建多线程的方式有两种:
1、继承QObject类 //更灵活,官方推荐
2、直接继承QThread类,通过QThread子类来创建线程 //有局限性

一、继承QObject方法实现多线程

//普通的一个Worker 类,包含自定义信号和槽class Worker : public QObject{Q_OBJECT//自定义槽函数public slots:void doWork(const QString &parameter) {QString result;/* ... here is the expensive or blocking operation ... */emit resultReady(result);}
//自定义信号signals:void resultReady(const QString &result);};
//Controller 为主线程中定义的类class Controller : public QObject{Q_OBJECTQThread workerThread;//直接创建一个线程public:Controller() {Worker *worker = new Worker;		//创建一个Worker 对象worker->moveToThread(&workerThread);//此语句非常重要。将创建的workerThread线程移到Worker对象中,这样worker 对象中就有一个新的名为workerThread的线程connect(&workerThread, &QThread::finished, worker, &QObject::deleteLater);//workerThread线程触发finished信号,将删除worker对象回收线程占用的资源。析构函数中就不需要使用delete worker;语句connect(this, &Controller::operate, worker, &Worker::doWork);//主线程的信号可以调用新添线程的槽函数connect(worker, &Worker::resultReady, this, &Controller::handleResults);//新添线程的信号可以触发主线程的槽函数workerThread.start();//启动新添线程}~Controller() {//新添的线程安全退出workerThread.quit();workerThread.wait();}//主线程槽函数public slots:void handleResults(const QString &);//主线程信号signals:void operate(const QString &);};

    Worker槽内的代码将在单独的线程中执行。可以自由地将Worker的插函数连接到任何信号,来自任何对象,任何线程。由于一种称为排队连接(queued connections)的机制,跨不同线程连接信号和插槽是安全的。

二、继承QThread方法实现多线程

    另一种使代码在单独线程中运行的方法是创建QThread的子类并重新实现QThread的run()函数。

试用情况:只适用于新添线程发出signal,主线程接收信号,触发主线程中实现的槽函数。
例如:

//继承QThread的WorkerThread 类只有自定义信号。因为WorkerThread 类对象可以发出信号class WorkerThread : public QThread{Q_OBJECT//重新实现run(),在run()中处理耗时耗空间的操作。避免主线程假死(如界面卡顿)void run() override {QString result;/* ... here is the expensive or blocking operation ... */emit resultReady(result);}signals:void resultReady(const QString &s);};void MyObject::startWorkInAThread(){WorkerThread *workerThread = new WorkerThread(this);//创建新workerThread 线程//新建workerThread 线程发出信号,主线程执行处理connect(workerThread, &WorkerThread::resultReady, this, &MyObject::handleResults);//新建workerThread线程发出结束信号,将调用deleteLater()删除workerThread 对象,回收新建线程资源connect(workerThread, &WorkerThread::finished, workerThread, &QObject::deleteLater);workerThread->start();}

    在该示例中,线程将在run函数返回后退出。除非调用exec(),否则线程中不会运行任何事件循环。

三、导致继承QThread的局限性原因

    QThread实例存在于实例化它的旧线程中,而不是在调用run()的新线程中。这意味着QThread的所有排队槽和调用的方法都将在旧线程中执行。因此,希望在新线程中调用槽的时候就只能使用继承QObject的方法;新的槽不应该直接实现到子类QThread中。
与排队槽或调用的方法不同,直接在QThread对象上调用的方法将在调用该方法的线程中执行。当子类化QThread时,请记住构造函数在旧线程中执行,而run()在新线程中执行。如果从两个函数访问成员变量,则从两个不同的线程访问该变量。检查这样做是否安全

补充说明:
    当线程是started()和finished()时,QThread将通过信号通知您,或者您可以使用isFinished()和isRunning()来查询线程的状态。
您可以通过调用exit()或quit()来停止线程。在极端情况下,您可以使用terminate()强制终止正在执行的线程。然而,这样做是危险和不鼓励的。
    从Qt 4.8开始,可以通过将finished()信号连接到QObject::deleteLater()来释放刚刚结束的线程中的对象。使用wait()来阻塞调用线程,直到另一个线程完成执行(或直到指定的时间过去)。

这篇关于Qt实现多线程 QThread的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/749825

相关文章

使用shardingsphere实现mysql数据库分片方式

《使用shardingsphere实现mysql数据库分片方式》本文介绍如何使用ShardingSphere-JDBC在SpringBoot中实现MySQL水平分库,涵盖分片策略、路由算法及零侵入配置... 目录一、ShardingSphere 简介1.1 对比1.2 核心概念1.3 Sharding-Sp

Java+AI驱动实现PDF文件数据提取与解析

《Java+AI驱动实现PDF文件数据提取与解析》本文将和大家分享一套基于AI的体检报告智能评估方案,详细介绍从PDF上传、内容提取到AI分析、数据存储的全流程自动化实现方法,感兴趣的可以了解下... 目录一、核心流程:从上传到评估的完整链路二、第一步:解析 PDF,提取体检报告内容1. 引入依赖2. 封装

Java实现复杂查询优化的7个技巧小结

《Java实现复杂查询优化的7个技巧小结》在Java项目中,复杂查询是开发者面临的“硬骨头”,本文将通过7个实战技巧,结合代码示例和性能对比,手把手教你如何让复杂查询变得优雅,大家可以根据需求进行选择... 目录一、复杂查询的痛点:为何你的代码“又臭又长”1.1冗余变量与中间状态1.2重复查询与性能陷阱1.

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

Redis实现分布式锁全过程

《Redis实现分布式锁全过程》文章介绍Redis实现分布式锁的方法,包括使用SETNX和EXPIRE命令确保互斥性与防死锁,Redisson客户端提供的便捷接口,以及Redlock算法通过多节点共识... 目录Redis实现分布式锁1. 分布式锁的基本原理2. 使用 Redis 实现分布式锁2.1 获取锁

Linux实现查看某一端口是否开放

《Linux实现查看某一端口是否开放》文章介绍了三种检查端口6379是否开放的方法:通过lsof查看进程占用,用netstat区分TCP/UDP监听状态,以及用telnet测试远程连接可达性... 目录1、使用lsof 命令来查看端口是否开放2、使用netstat 命令来查看端口是否开放3、使用telnet

使用SpringBoot+InfluxDB实现高效数据存储与查询

《使用SpringBoot+InfluxDB实现高效数据存储与查询》InfluxDB是一个开源的时间序列数据库,特别适合处理带有时间戳的监控数据、指标数据等,下面详细介绍如何在SpringBoot项目... 目录1、项目介绍2、 InfluxDB 介绍3、Spring Boot 配置 InfluxDB4、I

基于Java和FFmpeg实现视频压缩和剪辑功能

《基于Java和FFmpeg实现视频压缩和剪辑功能》在视频处理开发中,压缩和剪辑是常见的需求,本文将介绍如何使用Java结合FFmpeg实现视频压缩和剪辑功能,同时去除数据库操作,仅专注于视频处理,需... 目录引言1. 环境准备1.1 项目依赖1.2 安装 FFmpeg2. 视频压缩功能实现2.1 主要功

使用Python实现无损放大图片功能

《使用Python实现无损放大图片功能》本文介绍了如何使用Python的Pillow库进行无损图片放大,区分了JPEG和PNG格式在放大过程中的特点,并给出了示例代码,JPEG格式可能受压缩影响,需先... 目录一、什么是无损放大?二、实现方法步骤1:读取图片步骤2:无损放大图片步骤3:保存图片三、示php

使用Python实现一个简易计算器的新手指南

《使用Python实现一个简易计算器的新手指南》计算器是编程入门的经典项目,它涵盖了变量、输入输出、条件判断等核心编程概念,通过这个小项目,可以快速掌握Python的基础语法,并为后续更复杂的项目打下... 目录准备工作基础概念解析分步实现计算器第一步:获取用户输入第二步:实现基本运算第三步:显示计算结果进