PyQt5 解决界面无响应方案

文章目录

  • 前言
  • 版本
  • 案例
  • 解决方案
    • `QThread`
    • `QTimer`
  • 局部变量创建异步线程导致 UI 未响应
    • 如果 `QTimer 不使用 self.time` 写法
  • 个人简介

前言

  • 在PyQt5中,GUI线程通常指的是Qt的主事件循环线程,也称为主线程。主线程负责处理GUI事件、更新UI界面等任务。在PyQt5中,主线程和GUI线程是同一个线程,即运行应用程序的线程。
  • 当创建一个Qt应用程序时,主线程会启动,并执行QApplication.exec_()方法,进入Qt的事件循环。在事件循环中,主线程会不断地监听并处理用户的输入事件、定时器事件、网络事件等,然后更新UI界面。
  • 如果在主线程执行耗时操作,比如 循环、sleep、wait 异步线程执行 会导致 UI 界面进入无响应状态,我们可以采用以下两种方式异步处理:使用QThread 或 QTimer

版本

  • PyQt5
  • Python 3.x

案例

  • 我们写一个简单的进度条填充程序,每 2 秒填充 1%:
import sys
import timefrom PyQt5.QtWidgets import QApplication, QWidget, QProgressBar, QPushButton, QHBoxLayoutclass MyWidget(QWidget):def __init__(self):super(MyWidget, self).__init__()self.currentValue = 0self.progressBar = QProgressBar(self)self.progressBar.resize(200, 50)self.progressBar.move(20, 20)self.progressBar.setValue(self.currentValue)# 创建一个按钮self.button = QPushButton('点击我', self)self.button.clicked.connect(self.on_clicked)# 创建一个垂直布局,并将按钮添加到布局中layout = QHBoxLayout()layout.addWidget(self.progressBar)layout.addWidget(self.button)# 设置窗口的主布局为垂直布局self.setLayout(layout)def on_clicked(self):while True:time.sleep(2)self.currentValue = (self.currentValue + 1) % 101self.progressBar.setValue(self.currentValue)if __name__ == '__main__':app = QApplication(sys.argv)w = MyWidget()w.resize(500, 300)w.move(300, 300)w.setWindowTitle('Simple')w.show()sys.exit(app.exec_())
  • 点击运行,我们会发现 UI 界面出现无响应且进度条没有刷新:

解决方案

  • 为了避免 UI 界面无响应,我们可以采用以下两种方式:使用 QThread 或 QTimer

QThread

  • 我们可以通过点击事件创建 QThread 异步线程执行:
import sys
import timefrom PyQt5.QtCore import QThread, pyqtSignal
from PyQt5.QtWidgets import QApplication, QWidget, QProgressBar, QPushButton, QHBoxLayoutclass MyWorker(QThread):timeout = pyqtSignal()def __init__(self):super(MyWorker, self).__init__()def run(self):while True:time.sleep(2)self.timeout.emit()class MyWidget(QWidget):def __init__(self):super(MyWidget, self).__init__()self.worker = Noneself.currentValue = 0self.progressBar = QProgressBar(self)self.progressBar.resize(200, 50)self.progressBar.move(20, 20)self.progressBar.setValue(self.currentValue)# 创建一个按钮self.button = QPushButton('点击我', self)self.button.clicked.connect(self.on_clicked)# 创建一个垂直布局,并将按钮添加到布局中layout = QHBoxLayout()layout.addWidget(self.progressBar)layout.addWidget(self.button)# 设置窗口的主布局为垂直布局self.setLayout(layout)def on_clicked(self):self.worker = MyWorker()self.worker.timeout.connect(self.upgradeProgress)self.worker.start()def upgradeProgress(self):self.currentValue = (self.currentValue + 1) % 101self.progressBar.setValue(self.currentValue)if __name__ == '__main__':app = QApplication(sys.argv)w = MyWidget()w.resize(500, 300)w.move(300, 300)w.setWindowTitle('Simple')w.show()sys.exit(app.exec_())

运行效果:

QTimer

  • 我们可以通过点击事件创建 QTimer 定时器异步执行:
import sys
from PyQt5.QtCore import QTimer
from PyQt5.QtWidgets import QApplication, QWidget, QProgressBar, QPushButton, QHBoxLayoutclass MyWidget(QWidget):def __init__(self):super(MyWidget, self).__init__()self.currentValue = 0self.progressBar = QProgressBar(self)self.progressBar.resize(200, 50)self.progressBar.move(20, 20)self.progressBar.setValue(self.currentValue)# 创建一个按钮self.button = QPushButton('点击我', self)self.button.clicked.connect(self.on_clicked)# 创建一个垂直布局,并将按钮添加到布局中layout = QHBoxLayout()layout.addWidget(self.progressBar)layout.addWidget(self.button)# 设置窗口的主布局为垂直布局self.setLayout(layout)def on_clicked(self):# 定义一个定时器并启动定时器self.time = QTimer()self.time.timeout.connect(self.upgradeProgress)self.time.start(200)def upgradeProgress(self):self.currentValue = (self.currentValue + 1) % 101self.progressBar.setValue(self.currentValue)if __name__ == '__main__':app = QApplication(sys.argv)w = MyWidget()w.resize(500, 300)w.move(300, 300)w.setWindowTitle('Simple')w.show()sys.exit(app.exec_())
  • 运行效果:

局部变量创建异步线程导致 UI 未响应

  • 在使用 QThread 的案例中,将 on_clicked 方法改为如下写法,同样会导致 UI 未响应状态:
    def on_clicked(self):worker = MyWorker()worker.timeout.connect(self.upgradeProgress)worker.start()
  • 这是因为在Python中,类似于 worker = MyWorker() 这样的语句创建的对象在当前作用域中是局部变量,它的生命周期与当前作用域相关联。当当前作用域的代码执行完成后局部变量会被销毁。
  • 如果异步线程的任务还没有完成,而主线程的事件循环又需要等待任务完成才能继续执行,那么就会导致GUI线程无响应。这是因为主线程被阻塞在等待异步任务的过程中,无法处理事件。
  • 为了避免这种情况,我们应该将异步线程对象存储为实例变量(即使用 self.worker = MyWorker() ),这样可以确保异步线程对象的生命周期与主对象相同,直到异步任务完成。这样即使当前作用域的代码执行完成,异步线程仍然可以继续执行,并且主线程的事件循环也不会被阻塞。

如果 QTimer 不使用 self.time 写法

  • 同理,如果不使用 self.time 写法,会被当做当前作用域中的局部变量,当前作用域代码执行完成后就会被销毁,不再继续执行。

个人简介

👋 你好,我是 Lorin 洛林,一位 Java 后端技术开发者!座右铭:Technology has the power to make the world a better place.

🚀 我对技术的热情是我不断学习和分享的动力。我的博客是一个关于Java生态系统、后端开发和最新技术趋势的地方。

🧠 作为一个 Java 后端技术爱好者,我不仅热衷于探索语言的新特性和技术的深度,还热衷于分享我的见解和最佳实践。我相信知识的分享和社区合作可以帮助我们共同成长。

💡 在我的博客上,你将找到关于Java核心概念、JVM 底层技术、常用框架如Spring和Mybatis 、MySQL等数据库管理、RabbitMQ、Rocketmq等消息中间件、性能优化等内容的深入文章。我也将分享一些编程技巧和解决问题的方法,以帮助你更好地掌握Java编程。

🌐 我鼓励互动和建立社区,因此请留下你的问题、建议或主题请求,让我知道你感兴趣的内容。此外,我将分享最新的互联网和技术资讯,以确保你与技术世界的最新发展保持联系。我期待与你一起在技术之路上前进,一起探讨技术世界的无限可能性。

📖 保持关注我的博客,让我们共同追求技术卓越。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【ETAS CP AUTOSAR工具链】RTA-OS基本概念与开发流程

RTA-OS基于早期ETAS操作系统的成熟技术,迄今为止,已在全球超过3.5亿个ECU中使用。RTA-OS是一个可静态配置的抢占式实时操作系统(RTOS),它常被用于资源受限但有着高性能要求的方案中。内核的实现不仅遵循了AUTOSAR R3.x、R4.0、R4.1、R4.2、R4…

科学碳目标(SBTI)认证是什么?

科学碳目标(SBTI)认证是一种基于科学的减排目标认证和监测体系,旨在确保企业和国家制定的减排目标符合科学标准,并且能够实现全球气候目标的减缓效应。这个认证体系由全球碳项目和世界资源研究所(WRI)共同开…

视频号小店怎么做?店铺运营详细步骤讲解,全网独家

大家好,我是电商笨笨熊 视频号小店作为今年的电商黑马,下一个站在风口的项目,自是吸引了不少的玩家; 先不说视频号自身庞大的流量体系,单单是高客单的市场就值得尝试一把; 且当前视频号小店刚刚推出不久…

SolidWorks教育版本好用吗?

SolidWorks教育版本,作为一款专为教育领域设计的三维CAD软件,以其直观易用的界面、强大的功能以及丰富的教育资源,受到了广大师生的青睐。那么,SolidWorks教育版本好用吗?答案是肯定的。 首先,SolidWorks教…

【工具】如何提取一个mp4文件的关键帧

文章目录 怎么做如何安装ffmepgUbuntu 或 DebianCentOS 或 FedoramacOSWindows其他 Linux 发行版 实践什么是关键帧 怎么做 你可以使用ffmpeg这个强大的多媒体处理工具来提取mp4文件中的关键帧。以下是一个示例命令,可以使用ffmpeg从mp4文件中提取关键帧&#xff1…

网络编程基础回顾

计算机网络(5):运输层 OSI 模型与 TCP/IP 协议 OSI七层协议模型 (open system interconnection) 应用层:为应用数据提供服务表示层:数据格式转化,数据加密会话层:建立、维护和管理会话传输层&…

Jmeter性能测试(五)

一、Jmeter参数化常用方式 1、CSV 数据文件设置 2、查询数据库(JDBC Connection Configuration) 二、CSV 数据文件设置 1、准备一个txt文件(不需要写表头,直接写你要用的数据就行了,多个字段用英文逗号隔开) 2、添加一个CSV 数据文件设置(放全局最上…

PaddleOCR使用

最近在项目过程中需要用到文字识别的能力,之前没有接触过。需要对现有的开源能力进行调研和学习。 1. 基本概念 1.1 PaddlePaddle PaddlePaddle 是一个由百度开源,基于 Python 的深度学习框架。PaddlePaddle 针对不同的硬件环境提供了不同的安装包或安…

小丑的身份证和复印件 (BFS + Floyd)

本题链接:登录—专业IT笔试面试备考平台_牛客网 题目: 样例: 输入 2 10 (JOKERjoke #####asdr) 输出 12 思路: 根据题意,要求最短时间,实际上也可以理解为最短距离。 所以应该联想到有关最短距离的算法&…

第07-6章 应用层详解

HTTP、SSL:基于TCP,HTTP端口:80、HTTPS(加密)端口:443;FTP:基于TCP,两类端口:21、20(数据传输之前需要建立连接此时是21,真正传输数据时用20)TFTP…

07 常用工具集

本课时主要介绍常用的工具,将会讲解三个知识点: JVM 相关工具的作用和适用场景; Git 常用命令和工作流; Linux 系统中常用分析工具。 常用工具汇总 常用工具汇总如下图所示。 说明:这里列出的都是一些相对独立的工…

若依集成mybatis-plus 超详细教程(亲测可用)

文章目录 简介步骤第一步第二步第三步第四步第五步第六步 使用QueryWrapperservice层impl 实现接口类层Mapper层 简介 话不多说 直接跟着下面的教程操作,如果有报错私信我,或者通过博文下面的微信名片加我微信,免费解答哦! 步骤 …