Qt中多线程使用案列

news2024/12/23 13:27:18

Qt中多线程下载大文件

#pragma once

#include <QWidget>
#include <QPushButton>
#include "ThreadPool.h"
#include <QProgressBar>
#include <QLabel>
#include <QHBoxLayout>
#include <QVBoxLayout>
class MainWindow : public QWidget
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = Q_NULLPTR);

private:
   
	void initUI();

	void initConnect();

private:


	QPushButton*  m_dlBtn;
	DownLoad::ThreadPool  threadPool;
	QMap<QString ,std::pair<QLabel*,QProgressBar*> > controlMap;

};

#include "MainWindow.h"

#include "Task.h"
#include "ThreadPool.h"
MainWindow::MainWindow(QWidget *parent)
	: QWidget(parent)
{
	initUI();
	initConnect();
}

void MainWindow::initUI()
{
	m_dlBtn = new QPushButton(this);
	m_dlBtn->setText(QString("DownLoad"));
	QVBoxLayout*  layout = new QVBoxLayout();
	layout->setSpacing(10);
	layout->setContentsMargins(10, 10, 10, 10);
	layout->addWidget(m_dlBtn);
	


	for (int i = 0; i < 10; i++)
	{
		DownLoad::Task* task = new DownLoad::Task("http://mirrors.tuna.tsinghua.edu.cn/archlinux/iso/2023.12.01/archlinux-2023.12.01-x86_64.iso", QString("C:/Users/gd09861-hlw/Desktop/11111/archlinux-2023.12.01-x86_64_%1.iso").arg(i), DownLoad::Task::WorkModel::DOWNLOAD);
		threadPool.push(task);
		QLabel *label = new QLabel(this);
		label->setText(QString("%1").arg(i));
		QProgressBar *progressBar = new QProgressBar(this);
		controlMap.insert(task->id(), std::make_pair(label, progressBar));

		QHBoxLayout *hLayout = new QHBoxLayout;
		hLayout->addWidget(label);
		hLayout->addWidget(progressBar);
		layout->addLayout(hLayout);
	}
	this->setLayout(layout);
}

void MainWindow::initConnect()
{
	connect(m_dlBtn, &QPushButton::clicked, [&]() {
			
			threadPool.startAll();
		

		}


	);
	connect(&threadPool, &DownLoad::ThreadPool::sigUpdateTaskProgress, this, [&](QString id, qint64 bytesR, qint64 bytesT) {

		controlMap[id].second->setValue((bytesR*100.0f) / (bytesT*1.0f));
	
	});

	connect(&threadPool, &DownLoad::ThreadPool::sigUpdateTaskState, this, [&](QString id,DownLoad::Task::State state) {
		switch (state)
		{
		case DownLoad::Task::Start:
		{
			controlMap[id].second->setValue(0);
		}
			break;
		case DownLoad::Task::Stop: {
			controlMap[id].first->setText("Stop");
		}
			break;
		case DownLoad::Task::Finish:
			controlMap[id].first->setText("Finish");
			break;
		case DownLoad::Task::Error:
			controlMap[id].first->setText("error");
			break;
		default:
			break;
		}
	});
}

#ifndef  __TASK_QUEUE_H__
#define  __TASK_QUEUE_H__

#include <QObject>
#include <QString>
#include "Task.h"
#include <QQueue>
#include <QMutex>
namespace DownLoad {

#define  DEFAULT_THREAD_MAX_COUNT  3
	class ThreadPool :public QObject
	{
		Q_OBJECT
	public:


		ThreadPool();
		~ThreadPool();

		void init();
		void  push(Task *task);
		Task* pop();
		void startAll();
		void slotUpdateTaskState(QString id, Task::State  state);
		signals :
				void sigUpdateTaskProgress(QString id, qint64 bytesReceived, qint64 bytesTotal);
				void sigUpdateTaskState(QString id, Task::State  state);
	private:

		QQueue<Task*>  m_tasks;
	
		QList<QThread*>  m_threads;
	};
};
#endif
#include "ThreadPool.h"
#include <QMutexLocker>
#include <QThread>
#include "Task.h"
DownLoad::ThreadPool::ThreadPool()
{
	init();
}

DownLoad::ThreadPool::~ThreadPool()
{

}

void DownLoad::ThreadPool::init()
{
	for (int i = 0; i < DEFAULT_THREAD_MAX_COUNT; i++) {
		QThread  *thread = new QThread();
		m_threads.push_back(thread);
	}
}


void DownLoad::ThreadPool::push(Task *task)
{
	
	m_tasks.enqueue(task);
}

DownLoad::Task* DownLoad::ThreadPool::pop()
{
	
	return m_tasks.dequeue();
}

void DownLoad::ThreadPool::startAll()
{
	if (m_threads.isEmpty()) {
		return;
	}

	for (int i = 0; i < m_threads.count(); i++)
	{
		QThread *  thread = m_threads.at(i);
		if (thread->isRunning()) {
			continue;
		}
		if (m_tasks.isEmpty()) {
			return;
		}
		Task* task = pop();
		task->moveToThread(thread);
	
		connect(task, &Task::sigUpdateProgress, this, &ThreadPool::sigUpdateTaskProgress, Qt::QueuedConnection);
		connect(task, &Task::sigUpdateState, this, &ThreadPool::slotUpdateTaskState, Qt::QueuedConnection);
		connect(thread, &QThread::started, task, &Task::slotDoWork,Qt::QueuedConnection);
		connect(thread, &QThread::finished, task, &Task::deleteLater);
		thread->start();

	}
}

void DownLoad::ThreadPool::slotUpdateTaskState(QString id, Task::State  state)
{
	emit  sigUpdateTaskState(id, state);

	if (state == DownLoad::Task::Finish || state == DownLoad::Task::Error||state==DownLoad::Task::Stop) {
	
		startAll();
	}
}



#ifndef  __TASK_H__
#define  __TASK_H__

#include <QObject>
#include <QNetworkRequest>
#include <QNetworkReply>
#include <QNetworkAccessManager>
#include <QSharedPointer>
#include <QUrl>
#include <QString>
#include <QEventLoop>
#include <QMetaType>
namespace  DownLoad {
#define  DOWNLOAD_FILE_SUFFIX  ".tmp"

	

	class Task :public QObject
	{
		Q_OBJECT
	public:
			enum WorkModel
		{
			UPLOAD,
			DOWNLOAD,
		};

		enum State
		{
			Start,
			Stop,
			Finish,
			Error,
		};
	
		QString  id();
		Task(const QString &strUrl, const QString &filePath, const WorkModel& workModel);
		~Task();
		void setSupportBreakPoint(bool isSupport);
		QString lastError();
	signals:


		void sigUpdateState(QString id,State  state);
		void sigUpdateProgress(QString id,qint64 bytesReceived, qint64 bytesTotal);
		public slots:

		void slotDoWork();

		void slotStopWork();

		void slotCancelWork();


	protected:
		void removeTmpFile(const QString &filePath);
		void slotUpdateProgress(qint64 bytesReceived, qint64 bytesTotal);
		void slotWriteFile();
		void slotFinish();
		void slotError(QNetworkReply::NetworkError code);
	protected:
		void doDownWork();
		void doUploadWork();
	private:


		QNetworkReply*				 m_reply = nullptr;
		QNetworkRequest				 m_request;
		QNetworkAccessManager		 m_manager;
		QUrl m_url;
		QString m_filePath = "";
		WorkModel  m_workModel = UPLOAD;
		QSharedPointer<QEventLoop>  m_loop;
		bool m_bSupportBPoint = false;
		qint64  m_bytesReceived;
		qint64  m_bytesTotal;
		qint64  m_bytesCurrentReceived;
		QString m_error = "";
		QString m_id = "";
	};
	
};
Q_DECLARE_METATYPE(DownLoad::Task::State);

#endif     //__TASK_H__
#include "Task.h"
#include <QFileInfo>
#include <QUuid>
#include <QDebug>
#include <QThread>
#include <QDir>


QString DownLoad::Task::id()
{
	return m_id;
}

DownLoad::Task::Task(const QString &strUrl, const QString &filePath, const WorkModel& workModel)
	:QObject(nullptr), m_url(strUrl), m_filePath(filePath), m_workModel(workModel),m_bytesTotal(0)
	,m_bytesReceived(0),m_bytesCurrentReceived(0),m_bSupportBPoint(false),m_id(QUuid::createUuid().toString())
{

}



DownLoad::Task::~Task()
{
	if (m_reply) {
		m_reply->deleteLater();
	}

}



void DownLoad::Task::setSupportBreakPoint(bool isSupport)
{
	m_bSupportBPoint = isSupport;
}

QString DownLoad::Task::lastError()
{
	return m_error;
}

void DownLoad::Task::slotDoWork()
{
	qDebug() << "UUID:" << m_id << "TID:" << QThread::currentThreadId()<<"\t"<<m_filePath;
	if (m_url.isEmpty() || m_filePath.isEmpty()) {
		return;
	}
	switch (m_workModel)
	{
	case DownLoad::Task::UPLOAD:
		doUploadWork();
		break;
	case DownLoad::Task::DOWNLOAD:
		doDownWork();
		break;
	default:
		break;
	}
}



void DownLoad::Task::slotStopWork()
{
	m_bytesCurrentReceived += m_bytesReceived;
	if (m_reply) {
		disconnect(m_reply, 0, this, 0);
		m_reply->abort();
		m_reply->deleteLater();
		m_reply = nullptr;
		this->thread()->exit();
		emit  sigUpdateState(m_id,State::Stop);
	}
}

void DownLoad::Task::slotCancelWork()
{
	slotStopWork();
	m_bytesCurrentReceived = 0;
	m_bytesReceived = 0;
	m_bytesTotal = 0;
	removeTmpFile(m_filePath + DOWNLOAD_FILE_SUFFIX);
}

void DownLoad::Task::removeTmpFile(const QString &filePath)
{
	QFileInfo fileInfo(filePath);
	if (fileInfo.exists()) {
		QFile::remove(filePath);
	}
}

void DownLoad::Task::slotUpdateProgress(qint64 bytesReceived, qint64 bytesTotal)
{
	m_bytesReceived = bytesReceived;
	m_bytesTotal = bytesTotal;
	emit  sigUpdateProgress(m_id,m_bytesReceived + m_bytesCurrentReceived, m_bytesTotal + m_bytesCurrentReceived);
}

void DownLoad::Task::slotWriteFile()
{
	QFile file(m_filePath + DOWNLOAD_FILE_SUFFIX);
	QDir dir=QFileInfo(m_filePath + DOWNLOAD_FILE_SUFFIX).absoluteDir();
	if (!dir.exists()) {
		dir.mkpath(dir.absolutePath());
	}
	if (file.open(QIODevice::WriteOnly | QIODevice::Append)) {
		file.write(m_reply->readAll());
	}
	file.close();
}

void DownLoad::Task::slotFinish()
{
	QVariant  code = m_reply->attribute(QNetworkRequest::HttpStatusCodeAttribute);
	qDebug() << "Error Code:" << code.toInt();
	if (m_reply->error() == QNetworkReply::NoError) {
		QFileInfo fileInfo(m_filePath + DOWNLOAD_FILE_SUFFIX);
		if (fileInfo.exists()) {
			QFile::rename(m_filePath+DOWNLOAD_FILE_SUFFIX,m_filePath);
			this->thread()->exit();
			emit sigUpdateState(m_id, State::Finish);
		}
	}
	else { 
		m_error = m_reply->errorString();
		this->thread()->exit();
		emit sigUpdateState(m_id, State::Error);
	}
}

void DownLoad::Task::slotError(QNetworkReply::NetworkError code)
{
	if (code == QNetworkReply::NoError)
		return;
	slotStopWork();
	removeTmpFile(m_filePath + DOWNLOAD_FILE_SUFFIX);
	this->thread()->exit();
	emit	 sigUpdateState(m_id, State::Error);
	m_error = m_reply->errorString();
	

}

void DownLoad::Task::doDownWork()
{
	
	if (m_bytesCurrentReceived <= 0) {
		removeTmpFile(m_filePath + DOWNLOAD_FILE_SUFFIX);
	}
	QFileInfo  fileInfo(m_filePath + DOWNLOAD_FILE_SUFFIX);
	if (fileInfo.exists()) {
		m_bytesCurrentReceived = fileInfo.size();
	}
	QString strUrl = m_url.toString();

	m_request.setUrl(strUrl);
	if (m_bSupportBPoint) {
		QString strRange = QString("bytes=%1-").arg(m_bytesCurrentReceived);
		m_request.setRawHeader("Range", strRange.toLatin1());
	}

	m_reply = m_manager.get(m_request);
	connect(m_reply, &QNetworkReply::downloadProgress, this, &Task::slotUpdateProgress);
	connect(m_reply, &QNetworkReply::readyRead, this, &Task::slotWriteFile);
	connect(m_reply, &QNetworkReply::finished, this, &Task::slotFinish);
	connect(m_reply, SIGNAL(error(QNetworkReply::NetworkError code)), this,SLOT(slotError(QNetworkReply::NetworkError code)));
}

void DownLoad::Task::doUploadWork()
{
		
}

上处代码运行可以看出,QThread 中,每次调用start() 时 ,都会改变线程ID 在这里插入图片描述
因此,QT 的线程开启就是在创建线程,只不过其中含有事件循环机制。另外对于自定义类型,必须指定队列连接。
运行后在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Matlab论文插图绘制模板第131期—函数等高线图

在之前的文章中&#xff0c;分享了Matlab函数折线图的绘制模板&#xff1a; 函数三维折线图&#xff1a; 函数网格曲面图&#xff1a; 函数曲面图&#xff1a; 进一步&#xff0c;再来分享一下函数等高线图。 先来看一下成品效果&#xff1a; 特别提示&#xff1a;本期内容『数…

(9)Linux Git的介绍以及缓冲区

&#x1f4ad; 前言 本章我们先对缓冲区的概念进行一个详细的探究&#xff0c;之后会带着大家一步步去编写一个简陋的 "进度条" 小程序。最后我们来介绍一下 Git&#xff0c;着重讲解一下 Git 三板斧&#xff0c;一般只要掌握三板斧就基本够用了。 缓冲区&#xff…

PostgresSQL数据库中分区和分表的区别以及PostgresSQL创建表分区分表示例

1.分区分表理解 数据库分区和分表都是数据库中常用的数据分散存储技术&#xff0c;但它们的实现方式和应用场景有所不同。 分表&#xff1a;将一个大的表拆分成多个小的表&#xff0c;每个子表存储一部分数据。分表可以减轻单个表的数据量&#xff0c;提高查询效率&#xff0c…

系统架构设计师教程(八)系统质量属性与架构评估

系统质量属性与架构评估 8.1 软件系统质量属性8.1.1 质量属性概念开发期质量属性运行期质量属性 8.1.2 面向架构评估的质量属性8.1.3 质量属性场景描述 8.2 系统架构评估8.2.1 系统架构评估中的重要概念8.2.2 系统架构评估方法SAAM 方法ATAM方法CBAM 方法其他方法 8.3 ATAM方法…

C++继承与派生——(2)派生类

归纳编程学习的感悟&#xff0c; 记录奋斗路上的点滴&#xff0c; 希望能帮到一样刻苦的你&#xff01; 如有不足欢迎指正&#xff01; 共同学习交流&#xff01; &#x1f30e;欢迎各位→点赞 &#x1f44d; 收藏⭐ 留言​&#x1f4dd; 缺乏明确的目标&#xff0c;一生将庸庸…

内网渗透:拿下一台内网主机后如何继续攻击拿下域控主机

目录 #### 域的定义&#xff1a;将网络中的多台计算机在逻辑上组织到一起&#xff0c;进行集中管理。 域控攻击目标 1. 判断当前主机是否在域中 windows系统的三种登录方式 NTLM协议 从上篇文章拿下10.0.1.4这台服务器后&#xff0c;发展对内网进行攻击&#xff0c;打穿这…

[SWPUCTF 2021 新生赛]easy_md5

打开环境 这儿考弱比较&#xff0c;之前的WP里写过绕过方法&#xff0c;这儿我们直接数组绕过 构造GET传参name[]123 , POST传参password[]456得到flag

在区块链中看CHAT的独特见解

问CHAT&#xff1a;谈谈对区块链以及区块链金融的理解 CHAT回复&#xff1a;区块链是一种去中心化的分布式数据库技术&#xff0c;这种技术通过加密算法&#xff0c;使数据在网络中传输和存储的过程变得更加安全可靠。区块链的出现引领了存储、交易等形式的革命&#xff0c;改变…

SSH无密登陆配置

1 SSH介绍 ssh命令用于远程登录到其他计算机&#xff0c;实现安全的远程管理。 基本语法&#xff1a; ssh 域名/IP地址 示例&#xff1a; &#xff08;1&#xff09;从hadoop100服务器上远程连接hadoop101服务器 [hadoophadoop100 ~]$ ssh hadoop101 如果出现如下内容 Ar…

【科技前沿】数字孪生技术改革智慧供热,换热站3D可视化引领未来

换热站作为供热系统不可或缺的一部分&#xff0c;其能源消耗对城市环保至关重要。在双碳目标下&#xff0c;供热企业可通过搭建智慧供热系统&#xff0c;实现供热方式的低碳、高效、智能化&#xff0c;从而减少碳排放和能源浪费。通过应用物联网、大数据等高新技术&#xff0c;…

Mac如何搭建本地服务器

苹果电脑Mac OS X系统自带了Apache服务器 打开终端 //开启apache&#xff1a;sudo apachectl start //重启apache&#xff1a;sudo apachectl restart //关闭apache&#xff1a;sudo apachectl stop在浏览器输入127.0.10.1 &#xff0c; 如果页面出现 it works&#xff0c;则…

数据可视化分析大屏,大数据统计UI页面源文件(信息分析平台免费PS资料)

数据可视化可以帮助数据分析者更好地理解数据&#xff0c;发现数据中的规律和趋势。通过图表和图形等可视化工具&#xff0c;数据分析者可以更快速地发现数据中的关系&#xff0c;比如相关性、趋势、异常值等。对于普通用户来说&#xff0c;理解复杂的数据可能会很困难。通过数…

11 Vue3中v-bind绑定动态样式和动态样式类

概述 v-bind指令可以说是Vue3中最常用的指令之一&#xff0c;使用v-bind&#xff0c;我们几乎能够给任何实现动态的绑定比值。 这里&#xff0c;我们主要演示以下&#xff0c;通过v-bind动态绑定CSS样式。 基本用法 我们创建src/components/Demo11.vue&#xff0c;在这个组…

Android App程序应用未校验签名证书——————《风险等级高》

目录 应用签名未校验风险1、检测目的2、风险等级3、检测依据4、风险描述5、检测步骤6、结果描述7、解决方案7.1、Android 检验 APK 是否签名的代码7.2、检验APK签名 8、结尾 应用签名未校验风险 1、检测目的 检测App程序启动时是否校验签名证书。 防止App的盗版率。未进行签…

vscode打开多个标签页配置

前言 如果其中一个标签的文件没有修改&#xff0c;再打开一个文件时之前的打开的标签页就会被替换掉。 在工作中使用很不方便。 解决办法 文件-首选项--设置 下图取消勾选 取消之后如下 再去打开标签就会一致显示了

14:00面试,14:08就出来了,问的问题有点变态。。。

从小厂出来&#xff0c;没想到在另一家公司又寄了。 到这家公司开始上班&#xff0c;加班是每天必不可少的&#xff0c;看在钱给的比较多的份上&#xff0c;就不太计较了。没想到5月一纸通知&#xff0c;所有人不准加班&#xff0c;加班费不仅没有了&#xff0c;薪资还要降40%…

C++ Qt开发:StringListModel字符串列表映射组件

Qt 是一个跨平台C图形界面开发库&#xff0c;利用Qt可以快速开发跨平台窗体应用程序&#xff0c;在Qt中我们可以通过拖拽的方式将不同组件放到指定的位置&#xff0c;实现图形化开发极大的方便了开发效率&#xff0c;本章将重点介绍QStringListModel字符串映射组件的常用方法及…

【QT Visual Studio环境配置】error MSB8020: 无法找到 v141/v142 的生成工具(完整版)

首先要了解V**平台工具集根据你安装的Visual Studio版本不同而有所区别&#xff0c;知道这个就容易解决问题了&#xff0c;确定你安装的那个版本&#xff0c;需要使用哪个工具集。 v143–>VS2022v142–>VS2019v141–>VS2017v140–>VS2015v120–>VS2013 一、解决…

【网络编程】poll和epoll服务器的设计

文章目录 前言一、poll二、epoll 1.epoll初识2.epoll服务器的设计3.epoll的工作原理4.epoll的优点5.epoll的工作模式总结 前言 poll和select一样&#xff0c;也是一种linux中的多路转接的方案。而poll解决了select的两个问题&#xff1a; 1.select的文件描述符有上限的问题。…

05鸿蒙APP开发之加载网络列表

目录 1、概述2、http请求封装2.1、为什么要封装Http请求&#xff1f;2.2、封装后的网络请求 3、发起请求并渲染列表数据第一步&#xff1a;准备一个目前能用的&#xff0c;测试API地址&#xff0c;如下&#xff1a;第二步&#xff1a;创建对应的实体对象第三步&#xff1a;调用…