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

【PyQt】QThread快速创建多线程任务

pyqt通过QThread快速创建多线程任务

在 PyQt5 中使用多线程时,需要注意 GUI 线程(主线程) 和 工作线程 的分离。PyQt5 的主线程负责处理 GUI 事件,如果在主线程中执行耗时任务,会导致界面卡顿甚至无响应。因此,耗时任务应该放在工作线程中执行。

PyQt5 提供了 QThread 来实现多线程编程

1. 使用 QThread 的基本步骤

1.1创建一个继承自 QThread 的类
1.2重写 run() 方法,在 run() 中执行耗时任务
1.3在主线程中创建并启动工作线程。
1.4使用信号(pyqtSignal)实现线程间通信

2. 示例代码

以下是一个简单的示例,展示如何使用 QThread 在后台执行耗时任务,同时更新 GUI。

import sys
import time
from PyQt5.QtCore import QThread, pyqtSignal
from PyQt5.QtWidgets import QApplication, QWidget, QVBoxLayout, QPushButton, QLabel# 自定义工作线程
class WorkerThread(QThread):# 定义一个信号,用于更新 GUIupdate_signal = pyqtSignal(str)def run(self):for i in range(1, 6):time.sleep(1)  # 模拟耗时任务self.update_signal.emit(f"Working... {i}/5")  # 发送信号self.update_signal.emit("Task completed!")  # 任务完成# 主窗口类
class MainWindow(QWidget):def __init__(self):super().__init__()self.init_ui()def init_ui(self):self.setWindowTitle("PyQt5 多线程示例")self.setGeometry(100, 100, 300, 150)# 创建布局和控件layout = QVBoxLayout()self.label = QLabel("点击按钮开始任务", self)self.button = QPushButton("开始任务", self)layout.addWidget(self.label)layout.addWidget(self.button)self.setLayout(layout)# 绑定按钮点击事件self.button.clicked.connect(self.start_task)def start_task(self):self.label.setText("任务开始...")self.button.setEnabled(False)  # 禁用按钮,防止重复点击# 创建工作线程self.worker = WorkerThread()self.worker.update_signal.connect(self.update_label)  # 连接信号到槽函数self.worker.finished.connect(self.task_finished)  # 任务完成时触发self.worker.start()  # 启动线程def update_label(self, message):self.label.setText(message)  # 更新标签内容def task_finished(self):self.button.setEnabled(True)  # 启用按钮# 运行程序
if __name__ == "__main__":app = QApplication(sys.argv)window = MainWindow()window.show()sys.exit(app.exec_())

3. 代码说明

WorkerThread:
class WorkerThread(QThread):
# 定义一个信号,用于更新 GUIupdate_signal = pyqtSignal(str)def run(self):for i in range(1, 6):time.sleep(1)  # 模拟耗时任务self.update_signal.emit(f"Working... {i}/5")  # 发送信号self.update_signal.emit("Task completed!")  # 任务完成
  • 继承自 QThread,重写 run() 方法。

  • 在 run() 中执行耗时任务,并通过 pyqtSignal 发送信号更新 GUI

  • .emit() 是用于手动发射信号的方法。通常在自定义信号时使用,类似于clicked

MainWindow:
# 主窗口类
class MainWindow(QWidget):def __init__(self):super().__init__()self.init_ui()def init_ui(self):self.setWindowTitle("PyQt5 多线程示例")self.setGeometry(100, 100, 300, 150)# 创建布局和控件layout = QVBoxLayout()self.label = QLabel("点击按钮开始任务", self)self.button = QPushButton("开始任务", self)layout.addWidget(self.label)layout.addWidget(self.button)self.setLayout(layout)# 绑定按钮点击事件self.button.clicked.connect(self.start_task)
  • 主窗口类,包含一个标签和一个按钮。

  • 点击按钮后,启动工作线程并禁用按钮,防止重复点击。

  • 通过信号和槽机制更新标签内容。

信号与槽:

start_task是按钮点击之后的槽函数

def start_task(self):self.label.setText("任务开始...")self.button.setEnabled(False)  # 禁用按钮,防止重复点击# 创建工作线程self.worker = WorkerThread()self.worker.update_signal.connect(self.update_label)  # 连接信号到槽函数self.worker.finished.connect(self.task_finished)  # 任务完成时触发self.worker.start()  # 启动线程def update_label(self, message):self.label.setText(message)  # 更新标签内容def task_finished(self):self.button.setEnabled(True)  # 启用按钮
  • update_signal:用于从工作线程向主线程传递消息

  • finished:QThread 的内置信号,当 run() 方法执行完毕时触发。

4. 注意事项

线程安全:

不要在子线程中直接操作 GUI 元素(如更新标签、修改按钮状态等),必须通过信号和槽机制实现

PyQt5 的信号和槽机制是线程安全的。

避免阻塞主线程:

所有耗时任务都应放在工作线程中执行,确保主线程能够及时响应 GUI 事件。

线程生命周期:

线程启动后,run() 方法执行完毕时线程会自动结束

如果需要手动终止线程,可以调用 QThread 的 quit() 或 terminate() 方法(但不推荐强制终止线程)。

5. 更复杂的场景

如果需要处理更复杂的多线程任务(如线程池、任务队列等),可以使用 Python 的 concurrent.futures 模块或 QRunnable + QThreadPool 实现。

使用 QRunnable 和 QThreadPool 的示例
from PyQt5.QtCore import QRunnable, QThreadPool, pyqtSignal, QObjectclass Worker(QRunnable):def __init__(self):super().__init__()self.signals = WorkerSignals()def run(self):for i in range(1, 6):time.sleep(1)self.signals.update_signal.emit(f"Working... {i}/5")self.signals.update_signal.emit("Task completed!")class WorkerSignals(QObject):update_signal = pyqtSignal(str)# 在主窗口中使用
class MainWindow(QWidget):def __init__(self):super().__init__()self.init_ui()self.thread_pool = QThreadPool()  # 创建线程池def start_task(self):worker = Worker()worker.signals.update_signal.connect(self.update_label)self.thread_pool.start(worker)  # 将任务提交到线程池
线程池大小:

QThreadPool 默认会根据系统资源自动调整线程数量。

可以通过 QThreadPool.globalInstance().setMaxThreadCount() 设置最大线程数。

http://www.lryc.cn/news/527368.html

相关文章:

  • 智能码二维码的成本效益分析
  • 企业财务管理系统的需求设计和实现
  • Springboot集成Swagger和Springdoc详解
  • 类和对象(4)——多态:方法重写与动态绑定、向上转型和向下转型、多态的实现条件
  • ui-automator定位官网文档下载及使用
  • 董事会办公管理系统的需求设计和实现
  • ESP32和STM32在处理中断方面的区别
  • 零售业革命:改变行业的顶级物联网用例
  • 字符串算法笔记
  • 在Ubuntu上用Llama Factory命令行微调Qwen2.5的简单过程
  • ThinkPhp伪静态设置后,访问静态资源也提示找不到Controller
  • JavaScript赋能智能网页设计
  • 基于STM32的阿里云智能农业大棚
  • 80,【4】BUUCTF WEB [SUCTF 2018]MultiSQL
  • 深入探索imi框架:PHP Swoole的高性能协程应用实践
  • 【算法篇·更新中】C++秒入门(附练习用题目)
  • 对神经网络基础的理解
  • 存储基础 -- SCSI命令格式与使用场景
  • 从崩溃难题看 C 标准库与 Rust:线程安全问题引发的深度思考
  • 【CSS入门学习】Flex布局设置div水平、垂直分布与居中
  • 9. 神经网络(一.神经元模型)
  • R 语言 | future 包,非阻塞的执行耗时脚本
  • UE学习日志#12 Niagara特效大致了解(水文,主要是花时间读了读文档和文章)
  • 【数据结构】_链表经典算法OJ:合并两个有序数组
  • Mongodb副本集群为什么选择3个节点不选择4个节点
  • 基于 WEB 开发的手机销售管理系统设计与实现内容
  • LeetCode - Google 大模型校招10题 第1天 Attention 汇总 (3题)
  • Vue3 provide/inject用法总结
  • Linux——网络基础(1)
  • 【记录】日常|从零散记录到博客之星Top300的成长之路