Qt多线程技术【线程池】:QRunnable 和 QThreadPool

在现代软件开发中,尤其是在处理大量并发任务时,线程池技术是一种高效的解决方案。线程池不仅能提高程序的性能,还能有效管理线程的生命周期,避免频繁的线程创建和销毁所带来的性能损失。本文将以Qt中的 QThreadPoolQRunnable 为核心,通过具体代码实例来讲解线程池技术的应用及其工作原理。


线程池概述

线程池(ThreadPool)是一种用于管理和复用线程的技术。在多线程编程中,我们经常需要处理大量的小任务,频繁地创建和销毁线程会带来性能上的开销。线程池通过预先创建一定数量的线程来处理任务,任务完成后线程会被返回到线程池中等待下一次使用,从而避免了创建新线程的开销。

线程池可以根据任务量动态地调整线程的数量,保持一定数量的线程处于空闲状态,并且通过合理调度任务来提高并发执行的效率。

Qt为我们提供了 QThreadPoolQRunnable 类来轻松实现线程池机制。通过这两个类,开发者可以更简便地管理线程,并将复杂的并发任务拆分为小的可执行任务交给线程池去处理。


QRunnable 类解析

在Qt中,QRunnable 是一个用于表示线程池任务的基类。它并不像 QThread 那样直接创建和管理线程,而是通过将任务提交给 QThreadPool 来实现多线程工作。QRunnable 的主要作用是将任务封装成可执行的单元,每个 QRunnable 对象都会有一个 run() 方法,该方法定义了任务执行的具体操作。

QRunnable 提供了以下几个重要方法:

  • run(): 这是 QRunnable 类中的纯虚函数,用于定义任务的执行逻辑。开发者需要重写此方法,来描述任务的行为。
  • setAutoDelete(): 该方法允许在任务完成后自动删除该任务对象。这在使用 QThreadPool 时非常有用,可以避免内存泄漏。
  • setPriority(): 可以设置任务的优先级,QRunnable 支持通过此方法将任务分配不同的优先级。
QThreadPool 类解析

QThreadPool 类是Qt中的线程池实现类,负责管理并调度多个线程。QThreadPool 提供了线程池的创建、线程的管理和任务的调度等功能。开发者可以通过 QThreadPool 提交多个任务,并且线程池会自动分配线程来执行这些任务。

QThreadPool 提供了以下几个常用的方法:

  • globalInstance(): 返回一个全局的线程池实例,通常用于获取默认的线程池。
  • start(QRunnable *runnable): 向线程池中提交任务,线程池会根据当前线程的空闲情况分配线程来执行该任务。
  • waitForDone(): 阻塞等待线程池中的所有任务执行完成。这在某些场景下非常有用,例如需要确保所有任务都完成后再继续执行下一步操作。
  • setMaxThreadCount(): 设置线程池中最大线程数,防止系统资源过度消耗。

线程池技术的优势

  1. 减少开销:频繁创建和销毁线程会带来额外的开销。线程池通过复用线程,避免了这种性能浪费。
  2. 线程管理自动化:开发者不需要手动管理线程的创建、销毁等操作,线程池自动处理线程的生命周期。
  3. 避免资源浪费:通过动态调整线程池的大小,可以根据负载动态增加或减少线程数量,避免线程资源过度消耗。
  4. 高效并行执行:线程池可以同时执行多个任务,特别适用于需要处理大量短小任务的场景。

代码实现:使用 QThreadPoolQRunnable

为了更好地理解线程池的应用,我们提供了一个简单的Qt程序示例,展示如何使用 QThreadPoolQRunnable 类来处理并发任务。

头文件:worker.h
#ifndef WORKER_H
#define WORKER_H

#include <QRunnable>  // 用于创建线程任务
#include <QString>
#include <QDebug>
#include <QThread>


#define tc(a) QString::fromLocal8Bit(a)

// 工作任务类,继承自QRunnable
class Worker : public QRunnable
{
public:
    Worker(const QString &taskName, int retryCount = 3);  // 构造函数,传入任务名称和重试次数
    void run() override;  // 线程池中的任务执行逻辑

private:
    QString m_taskName;  // 任务名称
    int m_retryCount;    // 任务失败时的最大重试次数

    bool executeTask();  // 执行任务的模拟方法
};

#endif // WORKER_H
源文件:worker.cpp
#include "worker.h"
#include <QThread>
#include <QRandomGenerator>

// 构造函数,初始化任务名称和重试次数,设置任务自动删除
Worker::Worker(const QString &taskName, int retryCount)
    : m_taskName(taskName), m_retryCount(retryCount)
{
    setAutoDelete(true);  // 设置自动删除任务
}

// 重写run函数,线程池中的任务执行逻辑
void Worker::run()
{
    int attempt = 0;
    bool success = false;

    // 尝试执行任务,直到达到重试次数或任务成功
    while (attempt < m_retryCount && !success) {
        attempt++;
        qDebug() << tc("尝试执行任务:") << m_taskName << tc("尝试次数:") << attempt;
        success = executeTask();  // 执行任务
        if (!success) {
            qDebug() << tc("任务失败,重试中:") << m_taskName;
            QThread::sleep(2);  // 模拟任务失败后的等待
        }
    }

    if (success) {
        qDebug() << tc("任务完成:") << m_taskName;
    } else {
        qDebug() << tc("任务失败,超过最大重试次数:") << m_taskName;
    }
}

// 模拟任务执行的逻辑,50%概率失败
bool Worker::executeTask()
{
    // 使用随机数模拟任务失败
    return QRandomGenerator::global()->bounded(2) == 0;
}
主程序文件:main.cpp
#include <QCoreApplication>
#include <QThreadPool>
#include <QDebug>
#include "worker.h"

#define tc(a) QString::fromLocal8Bit(a)

int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);

    // 获取全局线程池实例
    QThreadPool *threadPool = QThreadPool::globalInstance();
    
    // 设置最大线程数为4
    threadPool->setMaxThreadCount(4);

    // 创建多个任务并添加到线程池
    Worker *task1 = new Worker(tc("任务 1"));
    Worker *task2 = new Worker(tc("任务 2"));
    Worker *task3 = new Worker(tc("任务 3"));
    Worker *task4 = new Worker(tc("任务 4"), 2);  // 设置任务4最大重试次数为2次
    Worker *task5 = new Worker(tc("任务 5"));

    // 向线程池中添加任务
    threadPool->start(task1);
    threadPool->start(task2);
    threadPool->start(task3);
    threadPool->start(task4);
    threadPool->start(task5);

    // 等待线程池中的任务完成
    threadPool->waitForDone();

    return a.exec();
}
尝试执行任务: 任务 1 尝试次数: 1
任务完成: 任务 1
尝试执行任务: 任务 2 尝试次数: 1
任务完成: 任务 2
尝试执行任务: 任务 3 尝试次数: 1
任务完成: 任务 3
尝试执行任务: 任务 4 尝试次数: 1
任务失败,重试中: 任务 4
尝试执行任务: 任务 4 尝试次数: 2
任务完成: 任务 4
尝试执行任务: 任务 5 尝试次数: 1
任务完成: 任务 5

代码讲解
  1. QRunnableQThreadPool 的结合使用

    • 我们通过继承 QRunnable 创建了一个 Worker 类来封装任务,每个 Worker 实例表示一个任务。
    • 任务的执行逻辑被定义在 run() 方法中,而任务的失败重试机制由 executeTask() 方法模拟。
    • main() 函数中,我们通过 QThreadPool::globalInstance() 获取全局线程池实例,设置最大线程数为4,并将多个任务提交到线程池执行。
  2. 自动删除任务

    • setAutoDelete(true) 确保任务在执行完成后自动被删除,这有效防止了内存泄漏。
  3. 任务执行过程

    • 每个任务都会随机失败,模拟实际应用中的网络请求或数据库操作失败的场景,最多重试3次。
    • QThread::sleep(2) 模拟任务执行时的延迟,使得任务的执行过程更加真实。
  4. 线程池管理

    • 线程池会根据当前可用线程的数量来调度任务,如果有空闲线程,任务会立刻执行;如果线程池的线程数已经达到最大值,新的任务会排队等待。

总结

线程池技术能够减少线程创建和销毁的开销,通过任务调度和线程复用提高了程序的性能和并发处理能力。QRunnable 提供了灵活的任务管理方式,QThreadPool 则负责高效的线程管理与任务调度。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/969079.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

链表 —— 常用技巧与操作总结详解

引言 链表作为一种动态数据结构&#xff0c;以其灵活的内存管理和高效的插入删除操作&#xff0c;在算法与工程实践中占据重要地位。然而&#xff0c;链表的指针操作复杂&#xff0c;容易引发内存泄漏和野指针问题。本文博主将从基础操作到高阶技巧&#xff0c;系统化解析链表的…

Renesas RH850 FDL库介绍

文章目录 FDL库(Data Flash Library)简介FDL库的核心功能FDL库的使用步骤关键注意事项示例应用场景总结FDL库(Data Flash Library)简介 FDL(Data Flash Library)是Renesas为RH850系列微控制器提供的数据闪存(Data Flash)操作库,用于简化数据闪存的擦除、写入、读取等…

Linux 配置 MySQL 定时自动备份到另一台服务器

Linux 配置 MySQL 定时自动备份到另一台服务器 前言1、配置服务器通信1.1&#xff1a;配置过程 2、编写自动备份sh脚本文件3&#xff1a;设置定时自动执行 前言 此方案可使一台服务器上的 MySQL 中的所有数据库每天 0 点自动转储为 .sql 文件&#xff0c;然后将文件同步到另一…

用php tp6对接钉钉审批流的 table 表格 明细控件 旧版sdk

核心代码 foreach ($flows[product_list] as $k>$gift) {$items_list[] [[name > 商品名称, value > $gift[product_name] ?? ],[name > 规格, value > $gift[product_name] ?? ],[name > 数量, value > $gift[quantity] ?? ],[name > 单位, v…

RV1126解码(1)

比如我们现在要拉一个流&#xff0c; 拉一个rtmp或者拉一个rtsp的流&#xff0c;让它显示到显示屏上面去&#xff0c;此时就要用到我们这个解码模块了&#xff0c;把它个解出来并且发到其他模块去。 主要功能是通过FFMPEG的API读取每一帧的音视频数据&#xff0c;并通过RV1126的…

sql:时间盲注和boolen盲注

关于时间盲注&#xff0c;boolen盲注的后面几个获取表、列、具体数据的函数补全 时间盲注方法 import time import requests# 获取数据库名 def inject_database(url):dataname for i in range(1, 20):low 32high 128mid (low high) // 2while low < high:payload &q…

DeepSeek+Excel 效率翻倍

2025年初&#xff0c;DeepSeek以惊人的效率突破技术壁垒&#xff0c;用极低的成本实现了与行业顶尖AI相媲美的性能&#xff0c;瞬间成为全球科技领域的热门话题。 那么AI工具的普及将如何改变我们的工作方式&#xff1f;Excel会被取代吗&#xff1f; 今天&#xff0c;珠珠带你…

WPS或word接入智能AI

DeepSeek接入WPS 配置WPS &#xff08;1&#xff09;下载 OfficeAl助手插件: 插件下载地址:https://www.office-ai.cn/。 安装插件后&#xff0c;打开WPS&#xff0c;菜单栏会新增"OfficeAl助手”选项卡。 如果没有出现&#xff0c; 左上找到文件菜单 -> 选项 ,在…

论文学习记录之《CLR-VMB》

目录 一、基本介绍 二、介绍 三、方法 3.1 FWI中的数据驱动方法 3.2 CLR-VMB理论 3.3 注意力块 四、网络结构 4.1 网络架构 4.2 损失函数 五、实验 5.1 数据准备 5.2 实验设置 5.3 训练和测试 5.4 定量分析 5.5 CLR方案的有效性 5.6 鲁棒性 5.7 泛化性 六、讨…

使用 EDOT 监测由 OpenAI 提供支持的 Python、Node.js 和 Java 应用程序

作者&#xff1a;来自 Elastic Adrian Cole Elastic 很自豪地在我们的 Python、Node.js 和 Java EDOT SDK 中引入了 OpenAI 支持。它们为使用 OpenAI 兼容服务的应用程序添加日志、指标和跟踪&#xff0c;而无需任何代码更改。 介绍 去年&#xff0c;我们宣布了 OpenTelemetry…

Golang的多团队协作编程模式与实践经验

Golang的多团队协作编程模式与实践经验 一、多团队协作编程模式概述 在软件开发领域&#xff0c;多团队协作编程是一种常见的工作模式。特别是对于大型项目来说&#xff0c;不同团队间需要协同合作&#xff0c;共同完成复杂的任务。Golang作为一种高效、并发性强的编程语言&…

Sequence to Sequence model

基础模型 基础模型是用RNN模型&#xff0c;前部分是encoder用来寻找法语输入的编码&#xff0c;后半部分是decoder用来生成英文翻译作为输出&#xff0c;每次输出一个单词&#xff0c;直到输出结束标志如EOS。 下面是另一个例子&#xff0c;在CNN模型输出层之前会输出图片的向…

verilog练习:i2c slave 模块设计

文章目录 前言1.结构2.代码2.1 iic_slave.v2.2 sync.v2.3 wr_fsm.v2.3.1 状态机状态解释 2.4 ram.v 3. 波形展示4. 建议5. 资料总结 前言 首先就不啰嗦iic协议了&#xff0c;网上有不少资料都是叙述此协议的。 下面将是我本次设计的一些局部设计汇总&#xff0c;如果对读者有…

【竞技宝】PGL瓦拉几亚S4预选:Tidebound2-0轻取spiky

北京时间2月13日,DOTA2的PGL瓦拉几亚S4预选赛继续进行,昨日进行的中国区预选赛胜者组首轮Tidebound对阵的spiky比赛中,以下是本场比赛的详细战报。 第一局: 首局比赛,spiky在天辉方,Tidebound在夜魇方。阵容方面,spiky点出了幻刺、火枪、猛犸、小强、巫妖,Tidebound则是拿到飞…

Android RenderEffect对Bitmap高斯模糊(毛玻璃),Kotlin(1)

Android RenderEffect对Bitmap高斯模糊(毛玻璃)&#xff0c;Kotlin&#xff08;1&#xff09; import android.graphics.Bitmap import android.graphics.BitmapFactory import android.graphics.HardwareRenderer import android.graphics.PixelFormat import android.graphic…

AI前端开发的崛起与ScriptEcho的助力

近年来&#xff0c;人工智能&#xff08;AI&#xff09;技术飞速发展&#xff0c;深刻地改变着软件开发的格局。尤其是在前端开发领域&#xff0c;AI的应用越来越广泛&#xff0c;催生了对AI写代码工具的需求激增&#xff0c;也显著提升了相关人才的市场价值。然而&#xff0c;…

【Mac排错】ls: command not found 终端命令失效的解决办法

【TroubleShooting on Mac】ls: command not found 终端命令失效的解决办法 A Solution to Solve “Command not found” of Terminal on Mac 一直在使用心爱的MacBook Pro的Terminal&#xff0c;并且为她定制了不同的Profile。 这样&#xff0c;看起来她可以在不同季节&…

DexVLA:通用机器人控制中具有插件式扩散专家的视觉语言模型

25年2月来自美的集团和华东师范的论文“DexVLA: Vision-Language Model with Plug-In Diffusion Expert for General Robot Control”。 让机器人能够在不同的环境中执行不同的任务是机器人学习的核心挑战。虽然视觉-语言-动作 (VLA) 模型已显示出可泛化机器人技能的前景&…

【微服务学习一】springboot微服务项目构建以及nacos服务注册

参考链接 3. SpringCloud - 快速通关 springboot微服务项目构建 教程中使用的springboot版本是3.x&#xff0c;因此需要使用jdk17&#xff0c;并且idea也需要高版本&#xff0c;我这里使用的是IDEA2024。 环境准备好后我们就可以创建springboot项目&#xff0c;最外层的项目…

DeepSeek 助力 Vue 开发:打造丝滑的返回顶部按钮(Back to Top)

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…