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

pyQt学习笔记——QThread线程

PyQt 线程教程:使用 QThread 进行多线程编程

    • 为什么要使用 QThread?
      • 解决方案:
    • 创建 QThread 线程
      • 继承 QThread 创建线程
    • 在 GUI 界面中使用 QThread
    • 线程间通信(Signal & Slot)
    • 线程安全的停止方法
    • 多线程处理摄像头视频流
    • 结束线程释放资源
    • 总结

在 PyQt 中,主线程负责 GUI 的事件循环,因此如果执行耗时任务(如视频处理、深度学习推理等)可能会导致界面卡顿甚至无响应。为了避免这个问题,我们通常使用 QThread 创建后台线程来处理这些任务。本教程将详细介绍 QThread 的使用,包括线程的创建、启动、通信以及正确的管理方式。

为什么要使用 QThread?

在 PyQt 中,默认所有操作都在 主线程(GUI 线程) 中执行。如果执行耗时任务(如读取摄像头视频、文件处理、深度学习推理等),就会阻塞 GUI,使界面卡顿甚至无响应。

解决方案:

  • 使用 QThread 创建后台线程,避免阻塞 GUI。
  • 通过信号和槽(Signal & Slot)进行线程间通信,确保数据传输安全。

创建 QThread 线程

在 PyQt 中,我们可以通过两种方式创建线程:

  1. 继承 QThread 并重写 run() 方法(推荐使用)
  2. 使用 moveToThread() 方式(适用于复杂对象)

这里主要介绍第一种方式。

继承 QThread 创建线程

from PyQt5.QtCore import QThread, pyqtSignal
import time

class WorkerThread(QThread):
    update_signal = pyqtSignal(str)  # 定义信号,向主线程发送字符串

    def __init__(self):
        super().__init__()
        self.is_running = True  # 控制线程是否运行

    def run(self):
        """线程执行的任务"""
        while self.is_running:
            time.sleep(1)  # 模拟耗时任务
            self.update_signal.emit("线程运行中...")  # 发送信号
        self.update_signal.emit("线程已停止")

    def stop(self):
        """停止线程"""
        self.is_running = False

在 GUI 界面中使用 QThread

from PyQt5.QtWidgets import QApplication, QWidget, QPushButton, QVBoxLayout, QLabel
from PyQt5.QtCore import Qt
import sys

class MainWindow(QWidget):
    def __init__(self):
        super().__init__()
        self.setWindowTitle("QThread 多线程示例")
        self.resize(300, 200)

        self.layout = QVBoxLayout()
        self.label = QLabel("点击按钮启动线程")
        self.button_start = QPushButton("启动线程")
        self.button_stop = QPushButton("停止线程")

        self.layout.addWidget(self.label)
        self.layout.addWidget(self.button_start)
        self.layout.addWidget(self.button_stop)
        self.setLayout(self.layout)

        # 创建线程
        self.thread = WorkerThread()
        self.thread.update_signal.connect(self.update_label)  # 连接信号
        
        # 绑定按钮事件
        self.button_start.clicked.connect(self.start_thread)
        self.button_stop.clicked.connect(self.stop_thread)

    def start_thread(self):
        if not self.thread.isRunning():
            self.thread.is_running = True
            self.thread.start()

    def stop_thread(self):
        self.thread.stop()

    def update_label(self, message):
        self.label.setText(message)

if __name__ == "__main__":
    app = QApplication(sys.argv)
    window = MainWindow()
    window.show()
    sys.exit(app.exec_())

线程间通信(Signal & Slot)

在 PyQt 中,QThread 不能直接更新 UI,需要通过 信号(Signal)和槽(Slot) 进行通信。

  • pyqtSignal:定义一个信号
  • connect():连接信号和槽
  • emit():在线程中发送信号

示例:

class WorkerThread(QThread):
    update_signal = pyqtSignal(str)

    def run(self):
        for i in range(5):
            time.sleep(1)
            self.update_signal.emit(f"线程运行中:{i}")  # 发送信号

线程安全的停止方法

在 PyQt5 中,不能直接使用 terminate() 结束线程,正确的方法是 使用标志位 控制 run() 方法。

class WorkerThread(QThread):
    def __init__(self):
        super().__init__()
        self.is_running = True

    def run(self):
        while self.is_running:
            time.sleep(1)
            print("线程运行中")
    
    def stop(self):
        self.is_running = False

多线程处理摄像头视频流

import cv2
import os
from PyQt5.QtCore import QThread, pyqtSignal

class CameraThread(QThread):
    frame_signal = pyqtSignal(object)  # 发送视频帧

    def __init__(self):
        super().__init__()
        self.cap = cv2.VideoCapture(0)  # 打开摄像头
        self.is_running = True  # 控制线程是否运行
    
    def run(self):
        while self.is_running:
            ret, frame = self.cap.read()
            if ret:
                self.frame_signal.emit(frame)  # 发送视频帧到 UI 进行显示
    
    def stop(self):
        self.is_running = False  # 停止线程循环
        self.cap.release()  # 释放摄像头资源

结束线程释放资源

if self.thread.isRunning():
    self.thread.stop()
    self.thread.quit()
    self.thread.wait()

总结

  • 使用 QThread 处理耗时任务,防止 GUI 卡死。
  • 通过 pyqtSignal 进行线程间通信,避免直接操作 UI。
  • 使用 is_running 变量安全停止线程,避免 terminate() 可能引发的问题。

适用场景:

  • 处理摄像头视频流(如 YOLO 目标检测)
  • 数据处理、文件下载等耗时任务
  • 后台计算任务,如深度学习推理

相关文章:

  • wordpress站标签也打不开网站产品怎么优化
  • 大淘客官网做的网站打不开如何网络营销
  • 青岛微网站开发加强服务保障满足群众急需m
  • 用axuer 做网站产品原型seo管理
  • 国外做螺栓比较好的网站广东省自然资源厅
  • 怎么在网站做支付端口对接/武汉疫情最新情况
  • Linux——文件(2)文件系统
  • Kafka - 消息零丢失实战
  • 深度学习笔记 | 漫游RNN(循环神经网络)
  • 天梯赛 L2-024 部落
  • [ISP] 理解白平衡的理想色温6500K
  • springboot—— Shiro实现认证和授权功能
  • webscanner漏洞扫描部署使用
  • HTML 表单:构建交互式网页的关键元素
  • 3D 地图渲染-区域纹理图添加
  • 库博静态代码分析工具Jenkins插件集成
  • Husky目标跟踪
  • Spring Boot集成Elasticsearch指南
  • idea清除git密码
  • C++ STL:六大组件全解析
  • 大数据(4.1)Hive架构设计与企业级实战:从内核原理到性能巅峰优化,打造高效数据仓库
  • Qt基本框架(2)
  • 强化学习经典策略梯度算法REINFORCE
  • CMake Presets教程
  • 开发一个小程序需要多久时间?小程序软件开发周期
  • 【Flask开发】嘿马文学web完整flask项目第2篇:2.用户认证,Json Web Token(JWT)【附代码文档】