本文主要讲解使用多线程模块QThread解决PyQt界面程序唉执行耗时操做时,程序卡顿出现的无响应以及界面输出没法实时显示的问题。用户使用工具过程当中出现这些问题时会误觉得程序出错,从而把程序关闭。这样,致使工具的用户使用体验很差。下面咱们经过模拟上述出现的问题并讲述使用多线程QThread模块解决此类问题的方法。多线程
使用PyQt界面程序,点击运行按钮后,程序在显示框中每秒打印1个数字。程序代码以下:app
# -*- coding: utf-8 -*- import sys import time from PyQt5.QtCore import QThread, pyqtSignal from PyQt5.QtWidgets import QApplication, QMainWindow from QThread_Example_UI import Ui_Form class MyMainForm(QMainWindow, Ui_Form): def __init__(self, parent=None): super(MyMainForm, self).__init__(parent) self.setupUi(self) self.runButton.clicked.connect(self.display) def display(self): for i in range(20): time.sleep(1) self.listWidget.addItem(str(i)) if __name__ == "__main__": app = QApplication(sys.argv) myWin = MyMainForm() myWin.show() sys.exit(app.exec_())
程序运行过程结果以下(点击Run按钮后界面出现未响应字样,同时程序也没有出现每隔1秒打印1个数字,实际结果是循环结束后20个数字一同展现):函数
问题分析工具
上述实现的GUI程序都是单线程运行,对于须要执行一个特别耗时的操做时就会出现该问题现象。要解决这种问题能够考虑使用多线程模块QThread。spa
QThread是Qt的线程类中最核心的底层类。因为PyQt的的跨平台特性,QThread要隐藏全部与平台相关的代码 要使用的QThread开始一个线程,能够建立它的一个子类,而后覆盖其它QThread.run()函数。线程
class Thread(QThread): def __init__(self): super(Thread,self).__init__() def run(self): #
接下来建立一个新的线程code
thread = Thread()
thread.start()
能够看出,PyQt的线程使用很是简单,创建一个自定义的类(如Thread),自我继承自QThread ,并实现其run()方法便可。在使用线程时能够直接获得Thread实例,调用其start()函数便可启动线程,线程启动以后,会自动调用其实现的run()的函数,该方法就是线程的执行函数 。orm
业务的线程任务就写在run()函数中,当run()退出以后线程就基本结束了,QThread有started和finished信号,能够为这两个信号指定槽函数,在线程启动和结束之时执行一段代码进行资源的初始化和释放操做,更灵活的使用方法是,在自定义的QThread实例中自定义信号,并将信号链接到指定的槽函数,当知足必定的业务条件时发射此信号。对象
start():启动线程blog
wait():阻止线程,直到知足以下条件之一
(1)与此QThread对象关联的线程已完成执行(即从run返回时),若是线程完成执行,此函数返回True,若是线程还没有启动,也返回True
(2)等待时间的单位是毫秒,若是时间是ULONG_MAX(默认值·),则等待,永远不会超时(线程必须从run返回),若是等待超时,此函数将会返回False
sleep():强制当前线程睡眠多少秒
QThread类中的经常使用信号
started:在开始执行run函数以前,从相关线程发射此信号
finished:当程序完成业务逻辑时,从相关线程发射此信号
先继承QThread类并从新实现其中的run()函数,也就是说把耗时的操做放入run()函数中。代码以下:
# -*- coding: utf-8 -*- import sys import time from PyQt5.QtCore import QThread, pyqtSignal from PyQt5.QtWidgets import QApplication, QMainWindow from QThread_Example_UI import Ui_Form class MyMainForm(QMainWindow, Ui_Form): def __init__(self, parent=None): super(MyMainForm, self).__init__(parent) self.setupUi(self) # 实例化线程对象 self.work = WorkThread() self.runButton.clicked.connect(self.execute) def execute(self): # 启动线程 self.work.start() # 线程自定义信号链接的槽函数 self.work.trigger.connect(self.display) def display(self,str): # 因为自定义信号时自动传递一个字符串参数,因此在这个槽函数中要接受一个参数 self.listWidget.addItem(str) class WorkThread(QThread): # 自定义信号对象。参数str就表明这个信号能够传一个字符串 trigger = pyqtSignal(str) def __int__(self): # 初始化函数 super(WorkThread, self).__init__() def run(self): #重写线程执行的run函数 #触发自定义信号 for i in range(20): time.sleep(1) # 经过自定义信号把待显示的字符串传递给槽函数 self.trigger.emit(str(i)) if __name__ == "__main__": app = QApplication(sys.argv) myWin = MyMainForm() myWin.show() sys.exit(app.exec_())
程序运行结果以下(实现了每隔1秒打印1个数字):
若是你实现的工具须要执行特别耗时的操做,能够参考使用本文多线程QThread处理方法实现。固然,工具实际实现过程当中的场景会比这复杂。好比,你的输出并非有固定时间间隔输出的文本框,能够尝试使用屡次self.trigger.emit(str)方法进行操做。