Qt5.14.2 深入理解Qt多线程编程,掌握线程池架构实现高效并发

本文主要是介绍Qt5.14.2 深入理解Qt多线程编程,掌握线程池架构实现高效并发,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


在高并发的软件系统中,多线程编程是解决性能瓶颈和提高系统吞吐量的有效手段。作为跨平台的应用程序开发框架,Qt为我们提供了强大的多线程支持。本文将深入探讨Qt多线程编程的实现细节,并介绍线程池的设计思想,帮助读者彻底掌握Qt多线程编程技巧。


一、Qt的两种多线程实现方式剖析

Qt中实现多线程编程主要有两种方式:重写QThread类的run()函数和使用信号与槽。


1、重写QThread的run()函数

这种方式需要继承QThread类并重写虚函数run(),将需要并发执行的代码逻辑放在run()函数中。例如:

class WorkThread : public QThread {
public:void run() override {//并发执行的代码qDebug() << "Current thread:" << QThread::currentThreadId();//执行耗时操作heavyWorkLoad();}
};

在主线程中,我们只需创建WorkThread对象并调用start()即可启动新线程:

WorkThread *worker = new WorkThread;
worker->start();

这种方法的优点是直观简单,缺点是run()函数作为线程执行体只能有一个入口,不太适合处理多个工作单元并发执行的场景。


2、使用信号与槽方式

Qt的信号与槽机制也可以用于实现多线程编程,它的思路是:

(1)、创建QThread对象作为新线程

(2)、创建执行体对象,并使用QObject::moveToThread()将其移动到新线程

(3)、在主线程通过连接信号与槽的方式,间接调用执行体对象的槽函数,从而启动新线程中的任务


在这里插入图片描述

具体代码如下:

//ExecutionBody.h
class ExecutionBody : public QObject {Q_OBJECT
public slots:void execution() {//并发执行的代码  qDebug() << "Executing in thread" << QThread::currentThreadId();heavyWorkLoad();}
};//main.cpp
int main() {QThread *worker = new QThread;ExecutionBody *body = new ExecutionBody;body->moveToThread(worker);QObject::connect(worker, &QThread::started, body, &ExecutionBody::execution);worker->start();return app.exec();
}

相比第一种方法,信号与槽方式支持在新线程中执行多个函数,更加灵活。但也相对复杂一些,开发者需要清晰地理解信号连接、事件循环等概念。


二、突破瓶颈,构建高效线程池

前面介绍了Qt的基本多线程实现方式,不过在实际项目中,如果只是简单地启动固定数量的线程,可能会面临以下问题:

(1)、线程的创建和销毁代价较高

(2)、线程数量太多,会加重系统的线程调度开销

(3)、大量线程空转,造成CPU资源浪费

为了解决这些问题,我们需要引入线程池的概念,将闲置的线程资源统一管理和调度,避免频繁创建和销毁线程。Qt提供了QThreadPool类实现了这一机制。


1、QThreadPool设计原理


QThreadPool内部管理了一组工作线程(工作者线程),当有任务投递时,线程池会将任务分配给空闲的工作线程执行,避免频繁创建和销毁线程。此外,QThreadPool还支持设置活跃线程数上限,在线程全部忙碌时也不会盲目创建新的工作线程,从而避免过度占用系统资源。

QThreadPool采用信号与槽的方式将任务分发给工作线程。具体来说,当我们调用QThreadPool::start()投递任务时,QThreadPool会为任务创建一个QRunnable对象,并通过内部信号连接到某个工作线程,由工作线程执行QRunnable的run()函数。

在这里插入图片描述


2、QThreadPool使用示例

下面通过一个简单的例子展示如何使用QThreadPool:

//WorkerTask.h
class WorkerTask : public QRunnable {
public:void run() override {//执行任务逻辑qDebug() << "Executing task in thread" << QThread::currentThreadId();heavyWorkLoad();}
};//main.cpp 
int main() {QThreadPool *pool = QThreadPool::globalInstance();//设置最大线程数pool->setMaxThreadCount(QThread::idealThreadCount());//投递任务for(int i=0; i<200; ++i) {  WorkerTask *task = new WorkerTask;pool->start(task);}return app.exec();
}

这个示例首先获取全局QThreadPool实例,并设置最大工作线程数为当前系统的理想线程数(通常为CPU核心数)。然后循环构建WorkerTask对象并调用QThreadPool::start()投递,线程池会自动将任务分发给空闲的线程执行。

需要注意的是,QThreadPool默认采用栈内存管理QRunnable对象,也就是说在QRunnable的run()函数执行完毕后,QThreadPool会自动销毁对象。如果我们需要在run()函数执行完毕后继续访问QRunnable对象的数据成员,应该设置QThreadPool的stackSize属性(即将对象放在堆内存分配)。


三、多线程开发中的注意事项

尽管QThreadPool大大简化了多线程编程流程,但在实际开发中,我们仍需注意一些潜在的安全隐患和性能风险:

1、线程间数据访问安全

当多个线程并发访问同一份数据时,很容易出现竞态条件。Qt提供了QMutex、QSemaphore、QReadWriteLock等同步原语类,我们可以利用它们来保护线程间共享数据的完整性。

另外,Qt还提供了QAtomicInteger和QAtomicPointer等原子操作类,能够确保基础数据类型的读写操作的原子性。对于简单的计数、状态位的读写,使用原子操作类可以避免加锁开销。


2、任务队列控制策略

使用QThreadPool虽然能避免频繁创建销毁线程,但如果任务投递过多且执行时间过长,任务队列会持续积压,可能导致响应延迟或内存占用過高。

因此,我们需要对任务队列的长度作出合理控制。QThreadPool提供了两个相关的API:

  • QThreadPool::reserveThread()可以为高优先级任务预留线程资源
  • QThreadPool::setMaxThreadCount()可以动态调整线程池的最大线程数

我们可以在投递任务前检查当前队列长度,对于优先级较高的任务使用reserveThread()保留资源,对于优先级较低的任务可以选择延迟投递或动态增加线程池大小。


3、避免死锁

在多线程编程中,如果多个线程互相持有对方所需要的锁资源,就会发生死锁。例如下面的代码:

QMutex mutex1, mutex2;//线程1
mutex1.lock(); 
...
mutex2.lock(); //阻塞//线程2  
mutex2.lock();
...
mutex1.lock(); //阻塞

避免死锁的一个常用策略是:对所有需要加锁的代码采用统一的加锁顺序,每个线程按相同顺序申请锁。


4、减少线程切换开销

线程切换是一个非常耗时的操作,会带来较大的性能开销。我们应该尽量减少线程切换的发生,例如:

  • 将密集计算的代码块集中在一个或几个线程中,避免在多个线程间切换
  • 避免线程中的循环中阻塞操作(如休眠、加锁等),这会使该线程长时间占用CPU
  • 采用无锁编程,利用原子操作和内存屏障指令实现线程安全操作

通过本文的介绍,希望你能够加深对Qt多线程编程的理解,在实际开发中合理使用多线程,提高应用程序的整体性能。下一篇文章将为你带来更多实战案例,进一步展示Qt多线程编程的实践技巧。

这篇关于Qt5.14.2 深入理解Qt多线程编程,掌握线程池架构实现高效并发的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

Java中的StringBuilder之如何高效构建字符串

《Java中的StringBuilder之如何高效构建字符串》本文将深入浅出地介绍StringBuilder的使用方法、性能优势以及相关字符串处理技术,结合代码示例帮助读者更好地理解和应用,希望对大家... 目录关键点什么是 StringBuilder?为什么需要 StringBuilder?如何使用 St

Python实现微信自动锁定工具

《Python实现微信自动锁定工具》在数字化办公时代,微信已成为职场沟通的重要工具,但临时离开时忘记锁屏可能导致敏感信息泄露,下面我们就来看看如何使用Python打造一个微信自动锁定工具吧... 目录引言:当微信隐私遇到自动化守护效果展示核心功能全景图技术亮点深度解析1. 无操作检测引擎2. 微信路径智能获

Java并发编程之如何优雅关闭钩子Shutdown Hook

《Java并发编程之如何优雅关闭钩子ShutdownHook》这篇文章主要为大家详细介绍了Java如何实现优雅关闭钩子ShutdownHook,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 目录关闭钩子简介关闭钩子应用场景数据库连接实战演示使用关闭钩子的注意事项开源框架中的关闭钩子机制1.

Python中pywin32 常用窗口操作的实现

《Python中pywin32常用窗口操作的实现》本文主要介绍了Python中pywin32常用窗口操作的实现,pywin32主要的作用是供Python开发者快速调用WindowsAPI的一个... 目录获取窗口句柄获取最前端窗口句柄获取指定坐标处的窗口根据窗口的完整标题匹配获取句柄根据窗口的类别匹配获取句

在 Spring Boot 中实现异常处理最佳实践

《在SpringBoot中实现异常处理最佳实践》本文介绍如何在SpringBoot中实现异常处理,涵盖核心概念、实现方法、与先前查询的集成、性能分析、常见问题和最佳实践,感兴趣的朋友一起看看吧... 目录一、Spring Boot 异常处理的背景与核心概念1.1 为什么需要异常处理?1.2 Spring B

Python位移操作和位运算的实现示例

《Python位移操作和位运算的实现示例》本文主要介绍了Python位移操作和位运算的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 位移操作1.1 左移操作 (<<)1.2 右移操作 (>>)注意事项:2. 位运算2.1

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

Qt实现网络数据解析的方法总结

《Qt实现网络数据解析的方法总结》在Qt中解析网络数据通常涉及接收原始字节流,并将其转换为有意义的应用层数据,这篇文章为大家介绍了详细步骤和示例,感兴趣的小伙伴可以了解下... 目录1. 网络数据接收2. 缓冲区管理(处理粘包/拆包)3. 常见数据格式解析3.1 jsON解析3.2 XML解析3.3 自定义

SpringMVC 通过ajax 前后端数据交互的实现方法

《SpringMVC通过ajax前后端数据交互的实现方法》:本文主要介绍SpringMVC通过ajax前后端数据交互的实现方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价... 在前端的开发过程中,经常在html页面通过AJAX进行前后端数据的交互,SpringMVC的controll