一、信号和槽介绍
信号(Signal)和槽(Slot)是Qt中的核心机制,也是在PyQt编程中对象之间进行通信的机制。在Qt中,每个QObject对象和PyQt中所有继承自QWidget的控件(这些都是QObject的子对象)都支持信号与槽机制。当信号发射时,连接的槽函数将会自动执行。在PyQt5中信号与槽通过object.signal.connect()方法连接。
PyQt的窗口控件类中有很多内置信号,开发者也可以添加自定义信号。信号与槽具有如下特点:
*一个信号可以连接多个槽
*一个信号可以连接另一个信号
*信号参数可以使任何Python类型
*一个槽可以监听多个信号
*信号与槽的连接方式可以是同步连接,也可以是异步连接
*信号与槽的连接可能会跨线程
*信号可能会断开
当事件或者状态发生改变时,就会发出信号。同时,信号会触发所有与这个事件(信号)相关的函数(槽)。信号与槽可以使多对多的关系。一个信号可以连接多个槽,一个槽也可以监听多个信号。
1、定义信号
PyQt的内置信号时自动定义的。使用PyQt5.QtCore.pyqtSIGNAL()函数可以为QObject创建一个信号,使用pyqtSingnal()函数可以把信号定义为类的属性。
1.1为QObject对象创建信号
使用pqtSignal()函数创建一个或多个重载的未绑定的信号作为类的属性,信号只能在QObject的子类中定义。信号必须在类创建时定义,不能在累创建后作为类的属性动态添加进来。types参数表示定义信号时参数的类型,name参数表示信号名字,该项缺省时使用类的属性名字。使用pyqtSignal()函数创建信号时,信号可以传递多个参数,并指定信号传递参数的类型,参数类型是标准的Python数据类型(字符串、日期、布尔类型、数字、列表、元组和字典)
1.2为控件创建信号
from PyQt5.QtCore import pyqtSignal
from PyQt5.QtWidgets import QMainWindow
class WinForm(QMainWindow):
btnClickedSignal = pyqtSignal()
上面代码为自定义控件WinForm创建了一个btnClickedSignal信号
2、操作信号
使用connect()函数可以把信号绑定到槽函数上。使用disconnect()函数可以解除信号与槽函数的绑定。使用emit()函数可以发射信号。
3、信号与槽的入门应用
信号与槽有三种使用方法,第一种是内置信号与槽的使用,第二种是自定义信号与槽的使用,第三种是装饰器的信号与槽的使用。
3.1内置信号与槽的使用
所谓内置信号与槽的使用,是指在发射信号时,使用窗口控件的函数,而不是自定义的函数。在信号与槽中,可以通过QObject.signal.connect将一个QObject的信号连接到另一个QObject的槽函数。例子如下:
""" | |
【简介】 | |
信号和槽例子 | |
""" | |
from PyQt5.QtWidgets import QPushButton , QApplication, QWidget | |
from PyQt5.QtWidgets import QMessageBox | |
import sys | |
app = QApplication(sys.argv) | |
widget = QWidget() | |
def showMsg(): | |
QMessageBox.information(widget, "信息提示框", "ok,弹出测试信息") | |
btn = QPushButton( "测试点击按钮", widget) | |
btn.clicked.connect( showMsg) | |
widget.show() | |
sys.exit(app.exec_()) |
这个例子将一个按钮对象的内置clicked信号连接到自定义的槽函数showMsg(),也可以说showMsg()函数响应了一个按钮单击事件。单击“测试点击按钮”按钮,就会弹出一个信息提示框。运行结果如下:
3.2自定义信号与槽的使用
指在发射信号时,不使用窗口控件的函数,而是使用自定义的函数(简单地说,就是使用pyqtSignal类实例发射信号)。之所以要使用自定义信号与槽,是因为通过内置函数发射信号有自身的缺陷。首先,内置函数值包含一些常用的信号,有些信号的发射找不到对应的内置函数;其次,只有在特定情况下(如按钮的点击事件)才能发射这种信号;最后,内置函数传递的参数是特定的,不可以自定义。使用自定义的信号函数则没有这些缺陷。
在PyQt5编程中,自定义信号与槽的适用范围很灵活,比如因为业务需求,在程序中的某个地方需要发射一个信号,传递多种数据类型(实际上就是传递参数),然后在槽函数中接收传递过来的数据,这样就可以非常灵活地实现一些业务逻辑。以下是Python风格的写法举例:
""" | |
【简介】 | |
内置信号槽示例 | |
""" | |
from PyQt5.QtCore import QObject, pyqtSignal | |
# 信号对象 | |
class QTypeSignal(QObject): | |
# 定义一个信号 | |
sendmsg = pyqtSignal(object) | |
def __init__(self): | |
super(QTypeSignal, self).__init__() | |
def run(self): | |
# 发射信号 | |
self.sendmsg.emit('Hello Pyqt5') | |
# 槽对象 | |
class QTypeSlot(QObject): | |
def __init__(self): | |
super(QTypeSlot, self).__init__() | |
# 槽对象里的槽函数 | |
def get(self, msg): | |
print("QSlot get msg => " + msg) | |
if __name__ == '__main__': | |
send = QTypeSignal() | |
slot = QTypeSlot() | |
# 1 | |
print('--- 把信号绑定到槽函数 ---') | |
send.sendmsg.connect(slot.get) | |
send.run() | |
# 2 | |
print('--- 把信号断开槽函数 ---') | |
send.sendmsg.disconnect(slot.get) | |
send.run() |
运行结果:
信号与槽连接的主要步骤如下:
(1)生成一个信号
sendmsg = pyqtSignal(object)
(2)将信号与槽函数绑起来
send.sendmsg.connect(slot.get)
(3)槽函数接收数据
def get(self,msg):
print(“QSlot get msg =>”+msg)
(4)发射信号的实现
self.sendmsg.emit('Hello Pyqt5')
(5)把信号绑定到槽对象中的槽函数get()上,所以槽函数能接收到所发射的信号-----字符串‘Hello Pyqt5’。至此,数据传递成功,就这么简单。
send = QTypeSignal()
slot = QTypeSlot()
print(‘---把信号绑定到槽函数上---’)
send.sendmsg.connect( slot.get)
send.run()
同理,断开信号与槽函数get()的连接,那么槽函数肯定就接收不到所发射的信号了
print(‘---把信号与槽函数的连接断开---’)
send.sendmsg.disconnect( slot.get)
send.run()
以上演示的是传递一个参数,如果要传递两个参数,也可以用这个思路,例子如下:
from PyQt5.QtCore import QObject , pyqtSignal | |
#信号对象 | |
class QTypeSignal(QObject): | |
#定义一个信号 | |
sendmsg = pyqtSignal( str,str) | |
def __init__( self): | |
super( QTypeSignal, self).__init__() | |
def run( self): | |
# 发射信号 | |
self.sendmsg.emit('第一个参数','第二个参数') | |
# 槽对象 | |
class QTypeSlot(QObject): | |
def __init__( self): | |