添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
PyQt5学习笔记(十二)多线程

PyQt5学习笔记(十二)多线程

多线程大一还没教到,咱也不敢瞎说。就谈谈本蒟蒻对多线程的一个直观的理解吧,可能会有贻笑大方之处,还请于评论区斧正,Thanks♪(・ω・)ノ(知道多线程的同学直接看下面的代码吧)。

假设我们编写了一个图书管理系统(应该许多人都做过这个课设吧),为了防止电脑突然宕机,我们可能存在这样的一种需求:要求程序自运行开始,每隔一段时间可以自动保存文件(将文件缓冲区的内容写入静态存储区),但是我们的程序是死板的、需要我们输入内容、按下回车键才能推动程序的进行,那怎么样才能做到即便是我不按任何按键,程序执行到任何一步,程序都能每隔一段时间自动保存呢?这时候就需要引入线程的概念,我们原本管理系统的那些代码就放在主线程执行,与主线程的时间线平行存在其余的线程,我们可以在其中一条线程上运行额外的、独立于主线程代码的保存文件程序。那么在不引起逻辑矛盾的前提下,两条线程同时运行, 程序并发地完成了管理系统的主任务和保存文件的辅助任务 。这便是我直观理解上的多线程。

除此之外,如果我们的主任务比较耗时,比如我们按下一个按钮触发一个很耗时的程序片段,如果放在主线程上运行,那么按钮按下半天都弹不起来,程序会于此发生 阻塞 。我们需要等好久才能执行下一步,是我们的工作效率大大降低。所以有时候 为了提高程序运行效率 ,我们需要开多个线程,将主线程上耗时的程序片段放在副线程上运行,而我们继续执行主线程的下一步。

总结下来就是,多线程 同时完成多个任务

希望不会误导各位和我一样的编程萌新,只要先建立一个感性认知就行。


动态显示当前时间(QTimer)

多线程的一个很直接的作用就是在我们的主窗口上动态地显示时间,PyQt5中提供了 QTimer 来开辟一个线程,可开启和关闭,并且可以每隔一段时间自动触发 timeout 信号。

import sys
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
class ShowTime(QWidget):
    # 用来记录已经创建的子窗口数,目的是为了新建的子窗口起名字
    count = 0
    def __init__(self):
        super(ShowTime, self).__init__()
        self.initUI()
    def initUI(self):
        self.setWindowTitle("动态显示当前时间")
        self.resize(500, 300)
        self.label = QLabel()
        self.startBtn = QPushButton("开始")
        self.endBtn = QPushButton("结束")
        layout = QGridLayout()
        self.timer = QTimer()
        # 每一秒自动发送timeout 信号(1秒是我们设定的interval)
        # 也就是说,showTime函数在QTime线程开始后会每隔1秒被调用
        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):
        # start方法会使得timer线程被开启
        # 并且我们设置每隔1000ms发送一次timeout信号
        self.timer.start(1000)
        # 因为我们已经按下了"开始"按钮,所以此处设置“开始“按钮不可被按下,“结束”按钮可以被按下
        # 这样就保证了“开始”与“结束”两个按钮,一个被按下时,另一个会弹起
        self.startBtn.setEnabled(False)
        self.endBtn.setEnabled(True)
    def endTimer(self):
        # 结束timer线程
        self.timer.stop()
        self.startBtn.setEnabled(True)
        self.endBtn.setEnabled(False)
if __name__ == '__main__':
    app = QApplication(sys.argv)
    main = ShowTime()
    main.show()
    sys.exit(app.exec_())

运行结果:

按下“开始”后的效果,每隔一秒,时间就会更新

总结一下(`・ω・´)。上述程序中,我们使用了 QTimer 创建了一个线程对象 timer ,这个线程对象有一个可被周期性激活的信号 timeout 。然后这个线程对象 timer 可以通过 start 方法开启线程,参数代表信号 timeout 被激活的周期(即指定每隔多少毫秒在新开辟的线程中释放 timeout 信号)。最后可以通过线程对象 timer stop() 方法结束线程。


让程序定时关闭

除了使用 QTimer start 方法开启一个周期性发送 timeout 信号的线程外,还可以使用 QTimer singleShot 静态方法开启一个线程:

QTimer.singleShot(1000, timeShow)

上面一句的意思就是执行到这一步后,程序会开辟一个线程,该线程会在1000ms后执行 timeShow() 函数(注意没括号)

这里需要额外提一遍Python中函数加括号和不加括号的区别(知道的同学直接看下面代码吧(`・ω・´))

  • 不带括号时,调用的是这个函数本身 ,是整个函数体,是一个函数对象,函数的返回值不会被填在原处
  • 带括号(参数或者无参),调用的是函数的执行结果,须等该函数执行完成的结果,函数本身代表函数的返回值

因为代码比较短,我们就是用函数式编程了,展示一下如何让程序自动关闭。

import sys
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
if __name__ == '__main__':
    app = QApplication(sys.argv)
    label = QLabel("<font color=red size=140><b>Hello World, 窗口在5秒后自动关闭!</b></font>")
    # 将窗口设置为无边框
    label.setWindowFlags(Qt.FramelessWindowHint)
    label.show()
    # 这一步开始,5000ms后app.quit()会被执行
    QTimer.singleShot(5000, app.quit)
    sys.exit(app.exec_())

运行结果:

窗体5秒后消失

使用线程类(QThread)编写计数器

除了上面那种使用 QTimer 的内置方法开辟线程来并发地完成任务之外,PyQt5还提供了线程类 QThread 来实现自定义的、更加灵活的多线程管理。

在讲原理之前,先介绍一个显示数码管风格数字的控件 QLCDNumber ,利用它的 display() 方法可以显示数码管风格的数字,类别上和 QLabel 一样,都是显示文本内容的一段标签:

if __name__ == '__main__':
    app = QApplication(sys.argv)
    lcdNumber = QLCDNumber()
    lcdNumber.display(2020)
    lcdNumber.show()
    lcdNumber.resize(500, 400)
    sys.exit(app.exec_())

运行效果:

七段数码管2020



好,我们接下来详细讲一下线程类的大致使用过程。首先我们需要独立于我们的控件类外定义一个 继承 QThread 的类,这个类代表我们自定义的工作线程 ,此处我就将我们自定义的工作线程称为 WorkThread

这个派生的类具有一个会被自动调用的函数 run ,我们需要重构这个函数。在这个函数中我们编写的内容就是这个工作线程干的事情,工作线程被开启后,这个函数会被自动调用。一般我们会 run 的主体写成一个死循环 ,来达到循环执行的目的(否则一般的指令都是一瞬间执行完的,开辟一个瞬间执行完指令的工作线程就失去了意义)。然后我们可以在死循环中加入 self.sleep() 来让工作线程在这停顿,注意该方法的单位是秒而不是毫秒。

在下面这个例子中我们还用到了自定义的信号 timer end 。自定义信号后面会讲到,但此处的步骤很简单,可以学会。先通过 pyqtSignal() 创建一个信号对象,该对象的变量名就是我们后面可以直接使用的信号,比如 timer = pyqtSignal() 创建了一个变量名为 timer 的信号,通过 self.timer.emit() 可以释放这信号。

import sys
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
sec = 0
# 编写工作线程
class WorkThread(QThread):
    timer = pyqtSignal()
    end = pyqtSignal()
    # 重构内部方法run
    def run(self):
        while True:
            self.sleep(1)       # 休眠1秒
            if sec == 5:
                self.end.emit() # 释放end信号
                break
            self.timer.emit()   # 发送timer信号
class Count(QWidget):
    def __init__(self):
        super(Count, self).__init__()
        self.initUI()
    def initUI(self):
        self.setWindowTitle("使用线程类(QThread)编写计数器")
        self.resize(300, 120)
        layout = QVBoxLayout()
        # 数码管控件
        self.lcdNumber = QLCDNumber()
        layout.addWidget(self.lcdNumber)
        button  = QPushButton("开始计数")
        layout.addWidget(button)
        # 创建我们自定义的工作线程
        self.workThread = WorkThread()
        # 通过点击按钮来启动工作线程
        button.clicked.connect(self.work)
        # 此处的timer和end信号都是我们在WorkThread自定义的信号
        # 注意观察它们两个的释放的逻辑
        self.workThread.timer.connect(self.countTime)
        self.workThread.end.connect(self.end)
        self.setLayout(layout)
    def countTime(self):
        global sec  # 将外部变量sec申明为全局变量
        sec += 1
        # 数码控件会显示sec存储的数字对应的数码管
        self.lcdNumber.display(sec)
    # 启动WorkThread工作线程
    def work(self):
        self.workThread.start()
    # 结束WorkThread工作线程
    def end(self):
        # 弹出一个消息对话框,参数为:self, 对话框名字, 对话框内容, 对话框按钮
        QMessageBox.information(self, "消息", "计数结束", QMessageBox.Ok)