Python Qt GUI设计:QTimer计时器类、QThread多线程类和事件处理类(基础篇—8)

一般情况下,应用程序都是单线程运行的,但是对于GUI程序来说,单线程有时候满足不了需求。例如,如果需要执行一个特别耗时的操作,在执行过程中整个程序就会卡顿,效果就非常不理想或者Windows系统也认为程序运行出错,自动关闭了程序。要解决这种问题就涉及多线程的知识。

一般来说,多线程技术涉及三种方法,其中第一种是使用计时器模块QTimer;第二种是使用多线程模块QThread;最后是使用事件处理的功能。

1、QTimer计时器类

如果要在应用程序中周期性地进行某项操作,比如周期性地检测主机的CPU值,则需要用到QTimer(定时器),QTimer类提供了重复的和单次的定时器。要使用定时器,需要先创建一个QTimer实例,将其timeout信号连接到相应的槽,并调用start()。然后,定时器会以恒定的间隔发出timeout信号。

当窗口控件收到timeout信号后,它就会停止这个定时器。这是在图形用户界面中实现复杂工作的一个典型方法,随着技术的进步,多线程在越来越多的平台上被使用,最终QTimer对象会被线程所替代。

QTimer类中的常用方法如下表所示:

QTimer类中的常用信号如下表所示:

通过一个示例,了解QTimer计时器类的使用方法,效果如下所示:

示例中,初始化一个定时器,把定时器的timeout信号与showTime()槽函数连接起来。使用连接的槽函数显示当前时间,并在标签上显示系统现在的时间。单击“开始"按钮,启动定时器,并使"开始"按钮失效。单击“结束"按钮,停止定时器,并使“结束"按钮失效。

实现代码如下所示:

from PyQt5.QtWidgets import QWidget,  QPushButton ,  QApplication ,QListWidget,  QGridLayout , QLabel

from PyQt5.QtCore import QTimer ,QDateTime

import sys

class WinForm(QWidget): 

def __init__(self,parent=None):

super(WinForm,self).__init__(parent)

self.setWindowTitle("QTimer demo")

self.listFile= QListWidget()

self.label = QLabel('显示当前时间')

self.startBtn = QPushButton('开始')

self.endBtn = QPushButton('结束')

layout = QGridLayout(self)

        # 初始化一个定时器

self.timer = QTimer(self)

        # showTime()方法

self.timer.timeout.connect(self.showTime)

layout.addWidget(self.label,0,0,1,2) 

layout.addWidget(self.startBtn,1,0)

layout.addWidget(self.endBtn,1,1)

self.startBtn.clicked.connect( self.startTimer)

self.endBtn.clicked.connect( self.endTimer)

self.setLayout(layout) 

def showTime(self):

# 获取系统现在的时间

time = QDateTime.currentDateTime()

# 设置系统时间显示格式

timeDisplay = time.toString("yyyy-MM-dd hh:mm:ss dddd");

# 在标签上显示时间

self.label.setText( timeDisplay )

def startTimer(self):

        # 设置计时间隔并启动

self.timer.start(1000)

self.startBtn.setEnabled(False)

self.endBtn.setEnabled(True)

def endTimer(self):

self.timer.stop()

self.startBtn.setEnabled(True)

self.endBtn.setEnabled(False)

if __name__ == "__main__": 

app = QApplication(sys.argv) 

form = WinForm() 

form.show() 

sys.exit(app.exec_())

2、QThread多线程类

QThread是Qt线程类中最核心的底层类,由于PyQt的跨平台特性,QThread要隐藏所有与平台相关的代码。

在使用线程时可以直接得到Thread实例,调用其start()函数即可启动线程。线程启动之后,会自动调用其实现的run方法,该方法就是线程的执行函数。

业务的线程任务就写在run()函数中,当run()退出之后线程基本就结束了。QThread有started和finished信号,可以为这两个信号指定槽函数,在线程启动和结束时执行一段代码进行资源的初始化和释放操作。更灵活的使用方法是,在自定义的QThread实例中自定义信号,并将信号连接到指定的槽函数,当满足一定的业务条件后发射此信号。

QThread类中的常用方法如下表所示:

QThread类中的常用信号如下表所示:

通过一个示例,了解QThread多线程类的使用方法,效果如下所示:

示例中,在主界面中有一个用于显示时间的LCD数字面板,还有一个用于启动任务的按钮。用户单击"测试"按钮后,将开始一次非常耗时的计算(在程序中用一个2000 000 000次的循环来模拟这次非常耗时的工作,在真实的程序中可能是一个网络下载操作,从网络上下载一个很大的视频文件),同时LCD数字面板开始显示所用的毫秒数,并通过一个计时器进行更新。但是单击”测试“按钮后可见窗口卡死无法操作。此时在PyQt中所有的窗口都在UI主线程中(就是执行了QApplication.exec()的线程),在这个线程中执行耗时的操作会阻塞UI线程,从而让窗口停止响应。如果窗口长时间没有响应,则会影响用户体验,更严重的会导致程序崩溃。所以,为了避免出现这样的问题,要使用QThread开启一个新的线程,在这个线程中完成耗时的操作。

实现代码如下所示:

import sys

from PyQt5.QtCore import *

from PyQt5.QtGui import *

from PyQt5.QtWidgets import *

global sec

sec=0

class WorkThread(QThread):

trigger = pyqtSignal()

def __int__(self):

super(WorkThread,self).__init__()

def run(self):

for i in range(2000000000):

pass

# 循环完毕后发出信号

self.trigger.emit()       

def countTime():

global  sec

sec += 1

# LED显示数字+1

lcdNumber.display(sec)         

def work():

# 计时器每秒计数

timer.start(1000) 

# 计时开始

workThread.start()     

# 当获得循环完毕的信号时,停止计数

workThread.trigger.connect(timeStop) 

def timeStop():

timer.stop()

print("运行结束用时",lcdNumber.value())

global sec

sec=0

if __name__ == "__main__": 

app = QApplication(sys.argv)

top = QWidget()

top.resize(300,120)

# 垂直布局类QVBoxLayout

layout = QVBoxLayout(top)

    # 加个显示屏   

lcdNumber = QLCDNumber()           

layout.addWidget(lcdNumber)

button = QPushButton("测试")

layout.addWidget(button)

timer = QTimer()

workThread = WorkThread()

button.clicked.connect(work)

    # 每次计时结束,触发 countTime

timer.timeout.connect(countTime)     

top.show()

sys.exit(app.exec_())

3、事件处理类

PyQt为事件处理提供了两种机制:高级的信号与槽机制以及低级的事件处理程序。本篇文博只介绍低级的事件处理程序即:processEvents()函数的使用方法,它的作用是处理事件,简单地说,就是刷新页面。

对于执行很耗时的程序来说,由于PyQt需要等待程序执行完毕才能进行下一步,这个过程表现在界面上就是卡顿。而如果在执行这个耗时程序时不断地运行

QApplication.processEvents(),那么就可以实现一边执行耗时程序,一边刷新页面的功能,给人的感觉就是程序运行很流畅。

因此QApplication.processEvents()的使用方法就是,在主函数执行耗时操作的地方,加入QApplication.processEvents()。

通过一个示例,了解事件处理类的使用方法,效果如下所示:

实现代码如下所示:

from PyQt5.QtWidgets import QWidget,  QPushButton ,  QApplication ,QListWidget,  QGridLayout

import sys

import time

class WinForm(QWidget): 

def __init__(self,parent=None):

super(WinForm,self).__init__(parent)

self.setWindowTitle("实时刷新界面例子")       

self.listFile= QListWidget()

self.btnStart = QPushButton('开始')

layout = QGridLayout(self)

layout.addWidget(self.listFile,0,0,1,2)

layout.addWidget(self.btnStart,1,1)

self.btnStart.clicked.connect( self.slotAdd)

self.setLayout(layout) 

def slotAdd(self):

for n in range(10):

str_n='File index {0}'.format(n)

self.listFile.addItem(str_n)

QApplication.processEvents()

time.sleep(1)

if __name__ == "__main__": 

app = QApplication(sys.argv) 

form = WinForm() 

form.show() 

sys.exit(app.exec_())

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,636评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,890评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,680评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,766评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,665评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,045评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,515评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,182评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,334评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,274评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,319评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,002评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,599评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,675评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,917评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,309评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,885评论 2 341

推荐阅读更多精彩内容