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