PyQt5 界面显示无响应的实现

作者:zulien 时间:2021-09-05 08:59:11 

在GUI程序中,主线程也叫GUI线程,因为它是唯一被允许执行GUI相关操作的线程。对于一些耗时的操作,如果放在主线程中,就是出现界面无法响应的问题。

界面假死分析

在编写QT的界面程序时,当我们调用QApplication.exec()时,我们就启动了QT的事件循环。在开始的时候,QT会发出一些事件来显示和绘制窗口部件。在这之后,事件循环就开始运行,不断地检查是不是有事件发生并且把这些事件发送给应用程序中的QObject。

当一个事件被处理时,其他事件也可能会产生并且追加到QT的事件队列中。如果我们在处理一个特定的事件上耗费过多的时间,用户界面就会变得不能够响应。例如在OCS保存一个观测流程的过程中,一直到文件保存完毕,窗口系统产生的一些事件才会被处
理。在保存过程中,这个应用程序就不能响应窗口系统的请求来重绘自己。

解决方法

  • 方式一使用多线程:一个处理应用程序用户界面的线程,另外一个执行文件保存的线程。

  • 方法二:调用QApplication.processEvents()

博主推荐使用第二种方法,该方法是在事件处理程序中调用QApplication.processEvents()。

这个函数告诉QT处理来处理任何没有被处理的事件,并且将控制权返回给调用者。实际上,QApplication.exec()就是一个不停调用QApplication.processEvents()函数的小while循环。这种方式的危险性在于,也许用户在观测流程未保存好之前就关闭了主窗口,或者在界面上通过鼠标或键盘执行了其它的输入,以至于观测流程未保存好就企图被程序使用。对于这个问题的解决办法是把 qApp -> processEvents(); 替换为 qApp -> eventLoop() -> processEvents( QEventLoop::ExcludeUserInput ); 通过这个调用告诉QT忽略鼠标和键盘事件。


...
 def downfile(self,file, url):
 print("开始下载:", file, url)
 try:
  r = requests.get(url, stream=True)
  with open(file, 'wb') as fd:
   for chunk in r.iter_content():
    fd.write(chunk)
    QApplication.processEvents()
 except Exception as e:
  print("下载失败了", e)
...

------------------------------------------补充一下方法一--------------------------》》》》》

说实话快有大半年没怎么使用过python了,关于多线程的处理方式,解释可能不是那么清楚。(目前是一个phper,上半年基本是补PHP方面的基础知识,也就是够用还不精通的一个状态)

先上一个半年前的小作品,是关于微信公众号方面的一些。

PyQt5 界面显示无响应的实现

这里就不谈用途与使用方法了,大概的讲一下,遇到界面假死的处理方法之一。话不多说,先上代码


from PyQt5.QtCore import QThread, pyqtSignal

class interface(QMainWindow, Ui_MainWindow):
"""
Class documentation goes here.
"""
def xxxx():
 "此处省略无数行代码......"
 self.Work()

def Work(self):
 self.thread = RunThread()
 self.thread.start()

class RunThread(QThread):
# python3,pyqt5与之前的版本有些不一样
# 通过类成员对象定义信号对象
# _signal = pyqtSignal(str)

trigger = pyqtSignal()

def __init__(self, parent=None):
 super(RunThread, self).__init__()

def __del__(self):
 self.wait()

def run(self):
 # 处理你要做的业务逻辑,这里是通过一个回调来处理数据,这里的逻辑处理写自己的方法
 dlg.Config['user'] = dlg.check_account['account']
 dlg.Config['passwd'] = dlg.check_account['password']
 dlg.Config['jk'] = 'http://xxx.com'
 if dlg.num != 1:
  dlg.operato.config_item(dlg.Config, dlg.wx_update) # 初始化配置
 else:
  dlg.operato.config_item(dlg.Config, dlg.wx_create) # 初始化配置

self.trigger.emit()

说实话还是蛮喜欢python的这种简洁的写法的,所以在很长的一段时间里,一直是比较注重代码的简洁度与良好的注释。em...,不过在其它语言中很难保持这种初心,现在是比较注重性能,响应时间,并发、安全等问题。

这里的interface是主窗口类,如果想在自己的窗口中实现,加一个RunThread类,并在主窗口中定义一个函数,用于调用Work类方法就可以了。通过代码可以看到,不到50行的代码就实现了方法一中的功能了。pyqt5有很多自己的方法,包括多线程等等。这里提供的是一种思路。当然还有很多种方式实现,大家可以去探索一下,好的方法可以一起分享讨论。

========================================7月24号更新=================================

先放一个效果图,

PyQt5 界面显示无响应的实现

正常情况下会将一些耗时函数扔进Qthread线程中来避免页面假死的情况。

但并不是所有的都是行的通的,

PyQt5 界面显示无响应的实现

当使用异步协程的时候,pyqt5推荐的是使用quamash


import sys
import asyncio
import time

from PyQt5.QtWidgets import QApplication, QProgressBar
from quamash import QEventLoop, QThreadExecutor

app = QApplication(sys.argv)
loop = QEventLoop(app)
asyncio.set_event_loop(loop) # NEW must set the event loop

progress = QProgressBar()
progress.setRange(0, 99)
progress.show()

async def master():
await first_50()
with QThreadExecutor(1) as exec:
 await loop.run_in_executor(exec, last_50)
# TODO announce completion?

async def first_50():
for i in range(50):
 progress.setValue(i)
 await asyncio.sleep(.1)

def last_50():
for i in range(50,100):
 loop.call_soon_threadsafe(progress.setValue, i)
 time.sleep(.1)

with loop: ## context manager calls .close() when loop completes, and releases all resources
loop.run_until_complete(master())

还有一种情况,就是在UI主线程中执行,需要注意的是,如果是耗时任务则会造成界面的卡死,并不大友好。

来源:https://blog.csdn.net/zulien/article/details/84990708

标签:PyQt5,界面,无响应
0
投稿

猜你喜欢

  • 基于javascript如何传递特殊字符

    2023-09-06 04:14:23
  • Python多进程分块读取超大文件的方法

    2023-12-07 02:19:28
  • 将Python的Django框架与认证系统整合的方法

    2022-05-09 20:33:15
  • 利用mycat实现mysql数据库读写分离的示例

    2024-01-12 21:55:52
  • asp如何制作一个WAP手机的WML网页?

    2010-06-29 21:21:00
  • python使用pip安装SciPy、SymPy、matplotlib教程

    2022-03-05 01:46:12
  • 如何在Python项目中引入日志

    2023-01-25 15:10:53
  • Golang使用协程实现批量获取数据

    2024-01-30 05:05:57
  • js实现砖头在页面拖拉效果

    2024-05-22 10:40:24
  • Centos6.5和Centos7 php环境搭建方法

    2023-11-14 19:25:38
  • Pygame 精准检测图像碰撞的问题

    2022-01-17 17:56:13
  • 解决Python传递中文参数的问题

    2021-04-10 09:00:07
  • SQL 中 NULL值测试代码

    2024-01-19 04:03:32
  • 对python条件表达式的四种实现方法小结

    2023-11-05 17:01:58
  • 基于Python 的进程管理工具supervisor使用指南

    2021-01-06 20:55:03
  • 详解利用django中间件django.middleware.csrf.CsrfViewMiddleware防止csrf攻击

    2023-03-16 14:33:51
  • python的exec、eval使用分析

    2022-05-26 18:38:25
  • asp base64加解密函数代码

    2011-03-31 11:02:00
  • python基础之错误和异常处理

    2021-07-28 05:46:15
  • Python基于文件内容实现查找文件功能

    2022-04-13 11:17:47
  • asp之家 网络编程 m.aspxhome.com