PyQt5 界面顯示無(wú)響應(yīng)的實(shí)現(xiàn)
在GUI程序中,主線程也叫GUI線程,因?yàn)樗俏ㄒ槐辉试S執(zhí)行GUI相關(guān)操作的線程。對(duì)于一些耗時(shí)的操作,如果放在主線程中,就是出現(xiàn)界面無(wú)法響應(yīng)的問(wèn)題。
界面假死分析
在編寫QT的界面程序時(shí),當(dāng)我們調(diào)用QApplication.exec()時(shí),我們就啟動(dòng)了QT的事件循環(huán)。在開(kāi)始的時(shí)候,QT會(huì)發(fā)出一些事件來(lái)顯示和繪制窗口部件。在這之后,事件循環(huán)就開(kāi)始運(yùn)行,不斷地檢查是不是有事件發(fā)生并且把這些事件發(fā)送給應(yīng)用程序中的QObject。
當(dāng)一個(gè)事件被處理時(shí),其他事件也可能會(huì)產(chǎn)生并且追加到QT的事件隊(duì)列中。如果我們?cè)谔幚硪粋€(gè)特定的事件上耗費(fèi)過(guò)多的時(shí)間,用戶界面就會(huì)變得不能夠響應(yīng)。例如在OCS保存一個(gè)觀測(cè)流程的過(guò)程中,一直到文件保存完畢,窗口系統(tǒng)產(chǎn)生的一些事件才會(huì)被處
理。在保存過(guò)程中,這個(gè)應(yīng)用程序就不能響應(yīng)窗口系統(tǒng)的請(qǐng)求來(lái)重繪自己。
解決方法
- 方式一使用多線程:一個(gè)處理應(yīng)用程序用戶界面的線程,另外一個(gè)執(zhí)行文件保存的線程。
- 方法二:調(diào)用QApplication.processEvents()
博主推薦使用第二種方法,該方法是在事件處理程序中調(diào)用QApplication.processEvents()。
這個(gè)函數(shù)告訴QT處理來(lái)處理任何沒(méi)有被處理的事件,并且將控制權(quán)返回給調(diào)用者。實(shí)際上,QApplication.exec()就是一個(gè)不停調(diào)用QApplication.processEvents()函數(shù)的小while循環(huán)。這種方式的危險(xiǎn)性在于,也許用戶在觀測(cè)流程未保存好之前就關(guān)閉了主窗口,或者在界面上通過(guò)鼠標(biāo)或鍵盤執(zhí)行了其它的輸入,以至于觀測(cè)流程未保存好就企圖被程序使用。對(duì)于這個(gè)問(wèn)題的解決辦法是把 qApp -> processEvents(); 替換為 qApp -> eventLoop() -> processEvents( QEventLoop::ExcludeUserInput ); 通過(guò)這個(gè)調(diào)用告訴QT忽略鼠標(biāo)和鍵盤事件。
... def downfile(self,file, url): print("開(kāi)始下載:", 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) ...
------------------------------------------補(bǔ)充一下方法一--------------------------》》》》》
說(shuō)實(shí)話快有大半年沒(méi)怎么使用過(guò)python了,關(guān)于多線程的處理方式,解釋可能不是那么清楚。(目前是一個(gè)phper,上半年基本是補(bǔ)PHP方面的基礎(chǔ)知識(shí),也就是夠用還不精通的一個(gè)狀態(tài))
先上一個(gè)半年前的小作品,是關(guān)于微信公眾號(hào)方面的一些。
這里就不談?dòng)猛九c使用方法了,大概的講一下,遇到界面假死的處理方法之一。話不多說(shuō),先上代碼
from PyQt5.QtCore import QThread, pyqtSignal class interface(QMainWindow, Ui_MainWindow): """ Class documentation goes here. """ def xxxx(): "此處省略無(wú)數(shù)行代碼......" self.Work() def Work(self): self.thread = RunThread() self.thread.start() class RunThread(QThread): # python3,pyqt5與之前的版本有些不一樣 # 通過(guò)類成員對(duì)象定義信號(hào)對(duì)象 # _signal = pyqtSignal(str) trigger = pyqtSignal() def __init__(self, parent=None): super(RunThread, self).__init__() def __del__(self): self.wait() def run(self): # 處理你要做的業(yè)務(wù)邏輯,這里是通過(guò)一個(gè)回調(diào)來(lái)處理數(shù)據(jù),這里的邏輯處理寫自己的方法 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()
說(shuō)實(shí)話還是蠻喜歡python的這種簡(jiǎn)潔的寫法的,所以在很長(zhǎng)的一段時(shí)間里,一直是比較注重代碼的簡(jiǎn)潔度與良好的注釋。em...,不過(guò)在其它語(yǔ)言中很難保持這種初心,現(xiàn)在是比較注重性能,響應(yīng)時(shí)間,并發(fā)、安全等問(wèn)題。
這里的interface是主窗口類,如果想在自己的窗口中實(shí)現(xiàn),加一個(gè)RunThread類,并在主窗口中定義一個(gè)函數(shù),用于調(diào)用Work類方法就可以了。通過(guò)代碼可以看到,不到50行的代碼就實(shí)現(xiàn)了方法一中的功能了。pyqt5有很多自己的方法,包括多線程等等。這里提供的是一種思路。當(dāng)然還有很多種方式實(shí)現(xiàn),大家可以去探索一下,好的方法可以一起分享討論。
========================================7月24號(hào)更新=================================
先放一個(gè)效果圖,
正常情況下會(huì)將一些耗時(shí)函數(shù)扔進(jìn)Qthread線程中來(lái)避免頁(yè)面假死的情況。
但并不是所有的都是行的通的,
當(dāng)使用異步協(xié)程的時(shí)候,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主線程中執(zhí)行,需要注意的是,如果是耗時(shí)任務(wù)則會(huì)造成界面的卡死,并不大友好。
到此這篇關(guān)于PyQt5 界面顯示無(wú)響應(yīng)的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)PyQt5 界面顯示無(wú)響應(yīng)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python實(shí)現(xiàn)無(wú)損放大圖片的示例代碼
這篇文章主要為大家詳細(xì)介紹了如何利用Python語(yǔ)言實(shí)現(xiàn)一個(gè)簡(jiǎn)單的無(wú)損放大圖片小程序,可以支持將JPG/PNG圖片無(wú)損放大上萬(wàn)像素,感興趣的可以了解一下2022-08-08Python爬蟲urllib和requests的區(qū)別詳解
這篇文章主要介紹了Python爬蟲urllib和requests的區(qū)別詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09基于python的圖片修復(fù)程序(實(shí)現(xiàn)水印去除)
這篇文章主要給大家介紹了關(guān)于python圖片修復(fù)程序的相關(guān)資料,可以用于實(shí)現(xiàn)圖片中水印去除,主要利用的是OpenCV這個(gè)框架實(shí)現(xiàn)的,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來(lái)一起看看吧2018-06-06關(guān)于命令行執(zhí)行Python腳本的傳參方式
這篇文章主要介紹了關(guān)于命令行執(zhí)行Python腳本的傳參方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-09-09Python進(jìn)行數(shù)據(jù)可視化Plotly與Dash的應(yīng)用小結(jié)
數(shù)據(jù)可視化是數(shù)據(jù)分析中至關(guān)重要的一環(huán),它能夠幫助我們更直觀地理解數(shù)據(jù)并發(fā)現(xiàn)隱藏的模式和趨勢(shì),本文主要介紹了Python進(jìn)行數(shù)據(jù)可視化Plotly與Dash的應(yīng)用小結(jié),具有一定的參考價(jià)值,感興趣的可以了解一下2024-04-04