多线程

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

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

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

1.动态显示当前时间(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信号)。最后可以通过线程对象timerstop()方法结束线程。

2.让程序定时关闭


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

QTimer.singleShot(1000, timeShow)

上面一句的意思就是执行到这一步后,程序会开辟一个线程,该线程会在1000ms后执行timeShow()函数(注意没括号)
这里需要额外提一遍Python中函数加括号和不加括号的区别(知道的同学直接看下面代码吧(`・ω・´))

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

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

import sys
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
import time


#
if __name__ == '__main__':
    app = QApplication(sys.argv)
    i = 3
    timer = QTimer()

    label = QLabel("<font color=red size=140><b>Hello World, 窗口在{}秒后自动关闭!</b></font>".format(i))
    # 将窗口设置为无边框
    label.setWindowFlags(Qt.FramelessWindowHint)
    label.show()
    # # 这一步开始,3000ms后app.quit()会被执行
    QTimer.singleShot(3000, app.quit)


    sys.exit(app.exec_())

运行结果:

窗体3秒后消失

3.使用线程类(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()来让工作线程在这停顿,注意该方法的单位是秒而不是毫秒。

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

import sys
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *

sec = 600

# 编写工作线程
class WorkThread(QThread):
    timer = pyqtSignal()
    end = pyqtSignal()


    # 重构内部方法run
    def run(self):
        while True:
            self.sleep(1)  # 休眠1秒
            if sec == 0:
                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()
        self.lcdNumber.display(sec)
        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)


if __name__ == '__main__':
    app = QApplication(sys.argv)
    main = Count()
    main.show()
    sys.exit(app.exec_())

运行结果:

600s倒数计时器

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值