python主界面无响应_PyQt5 界面显示无响应的实现

在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方面的基础知识,也就是够用还不精通的一个状态)

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

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

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号更新=================================

先放一个效果图,

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

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

当使用异步协程的时候,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主线程中执行,需要注意的是,如果是耗时任务则会造成界面的卡死,并不大友好。

到此这篇关于PyQt5 界面显示无响应的实现的文章就介绍到这了,更多相关PyQt5 界面显示无响应内容请搜索python博客以前的文章或继续浏览下面的相关文章希望大家以后多多支持python博客!

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值