PySide6 多线程信号传递

本人是一位PySide的初学者,最近刚有点整明白Pyside6的多线程间的信号传递,于是想记录一下。
之前在网上找资料时,感觉很多教程的代码都比较复杂,于是想到写一个简单版本的,也希望能给其他人一些参考。
由于本人也是初学者,因此可能有很多错误,也请大家不吝赐教。

演示

Demo

首先写一个主窗口:

from PySide6.QtCore import Signal,Slot,Qt,QThread
from PySide6.QtWidgets import QWidget,QVBoxLayout,QPushButton,QLabel,QApplication
import sys,time

class mainWindow(QWidget):
    def __init__(self) -> None:
        super().__init__()

        self.label = QLabel("Hello!")
        self.label.setAlignment(Qt.AlignCenter)
        self.but = QPushButton("Click!")
        
        self.layout = QVBoxLayout()
        self.layout.addWidget(self.label)
        self.layout.addWidget(self.but)
        self.setLayout(self.layout)
        self.setWindowTitle('Signal Example')
        self.resize(300,300)
        self.show()

        
if __name__ == '__main__':
    app = QApplication([])
    widgets = mainWindow()
    sys.exit(app.exec())

这一步应该不用多讲,如果看不懂建议从PySide基础开始学起。
主窗口

写一个线程

该线程主要就是倒数。我们要让主程序的窗口上现实剩余时间,就是Hello!的那个位置。

class Th(QThread):
    timer = Signal(int)
    finish = Signal(bool)
    def __init__(self) -> None:
        super().__init__()
    
    def run(self):
        # ptvsd.debug_this_thread()
        print('Start Timer')
        self.finish.emit(False)
        for x in range(5):
            self.timer.emit(5-x)
            time.sleep(1)
        self.finish.emit(True)

在这里,我用Signal初始化两个信号,用于传递剩余时间和是否完成倒计时。
当倒计时开始时,设置未完成,将False给finish信号:
self.finish.emit(False)
随后每秒,都会把剩余时间给timer信号:
self.timer.emit(5-x)
当倒计结束时,将True给finish信号:
self.finish.emit(True)

为主窗口添加槽并绑定

首先,我们点击Click!按钮应该开始倒计时,即执行我们上一步写的线程。

    @Slot()
    def fun(self):
        self.th = Th()
        self.th.timer.connect(self.flushlabel)
        self.th.finish.connect(self.isFinish)
        self.th.start()

初始化线程,注意这里要用类的成员变量,不要用局部变量:
self.th = Th()
之后是开始这个线程:
self.th.start()

信号

self.th.timer.connect(self.flushlabel)
self.th.finish.connect(self.isFinish)

这两个是完成信号与槽的绑定。
timer和finish都是上一步我们在倒计时的类中设定的信号。
我们希望,在Th类(倒计时的类)中,每次信号的emit,都会运行一次绑定的槽(self.flushlabel与self.isFinish)

槽函数

上一步绑定的槽,在这里我们来实现它。实现的代码是在主窗口类中的。

class mainWindow(QWidget):
    def __init__(self) -> None:
		
		...
		...
		...
		
    @Slot()
    def fun(self):
		
		...
		...
		...

    @Slot(int)
    def flushlabel(self,nu):
        self.label.setText(str(nu))
    @Slot(bool)
    def isFinish(self,bo):
        if bo is True:
            self.but.setEnabled(True)
        else:
            self.but.setEnabled(False)

@Slot(int)@Slot(bool)为标识槽函数的参数类型,不写似乎也没有问题。

    def flushlabel(self,nu):
        self.label.setText(str(nu))

这里。该函数用来获得timer每次emit的倒计时,将这个数字现实到主窗口的label中。

    def isFinish(self,bo):
        if bo is True:
            self.but.setEnabled(True)
        else:
            self.but.setEnabled(False)

这个函数来获得每次finish每次emit的状态,判断倒计时是否完成,用来控制主窗口的按钮是否可以点击。

至此,我们的槽函数与信号的绑定完成了。这样每次在子线程(倒计时)的信号被emit时,我们的主线程(主窗口)都能拿到其emit的内容,并且使用其内容对主窗口进行相应更改。

完整代码

import sys,time
from PySide6.QtCore import Signal,Slot,Qt,QThread
from PySide6.QtWidgets import QWidget,QVBoxLayout,QPushButton,QLabel,QApplication

class mainWindow(QWidget):
    def __init__(self) -> None:
        super().__init__()

        self.label = QLabel("Hello!")
        self.label.setAlignment(Qt.AlignCenter)
        self.but = QPushButton("Click!")
        self.but.clicked.connect(self.fun)
        
        self.layout = QVBoxLayout()
        self.layout.addWidget(self.label)
        self.layout.addWidget(self.but)
        self.setLayout(self.layout)
        self.setWindowTitle('Signal Example')
        self.resize(300,300)
        self.show()
    @Slot()
    def fun(self):
        self.th = Th()
        self.th.timer.connect(self.flushlabel)
        self.th.finish.connect(self.isFinish)
        self.th.start()

    @Slot(int)
    def flushlabel(self,nu):
        self.label.setText(str(nu))
    @Slot(bool)
    def isFinish(self,bo):
        if bo is True:
            self.but.setEnabled(True)
        else:
            self.but.setEnabled(False)

class Th(QThread):
    timer = Signal(int)
    finish = Signal(bool)
    def __init__(self) -> None:
        super().__init__()
    
    def run(self):
        print('Start Timer')
        self.finish.emit(False)
        for x in range(5):
            self.timer.emit(5-x)
            time.sleep(1)
        self.finish.emit(True)
        
if __name__ == '__main__':
    app = QApplication([])
    widgets = mainWindow()
    sys.exit(app.exec())
posted @ 2022-03-20 16:19  Rogunt  阅读(3390)  评论(1编辑  收藏  举报