赞
踩
信号(Signal)和槽((Slot)是Qt中的核心机制,也是在PyQt编程中对象之间进行通信的机制。在Qt中,每一个QObject对象和 PyQt中所有继承自QWidget的控件(这些都是QObject的子对象)都支持信号与槽机制。当信号发射时,连接的槽函数将会自动执行。在 PyQt 5中信号与槽通过object.signal.connect()方法连接。
PyQt的窗口控件类中有很多内置信号,开发者也可以添加自定义信号。信号与槽具有如下特点:
在GUI编程中,当改变一个控件的状态时(如单击了按钮),通常需要通知另一个控件,也就是实现了对象之间的通信。在早期的GUI编程中使用的是回调机制,在Qt中则使用一种新机制-信号与槽。在编写一个类时,要先定义该类的信号与槽,在类中信号与槽进行连接,实现对象之间的数据传输。
信号与槽机制示意图:
当事件或者状态发生改变时,就会发出信号。同时,信号会触发所有与这个事件(信号)相关的函数(槽)。信号与槽可以是多对多的关系。一个信号可以连接多个槽,一个槽也可以监听多个信号。
PyQt的内置信号是自动定义的。使用 PyQt5.QtCore.pyqtSignal()函数可以为Q0bject创建一个信号,使用 pyqtSingnal()函数可以把信号定义为类的属性。
官方文档介绍:
为QObject对象创建信号
使用pyqtSignal()函数创建一个或多个重载的未绑定的信号作为类的属性,信号只能在Q0bject的子类中定义。
信号必须在类创建时定义,不能在类创建后作为类的属性动态添加进来。types参数表示定义信号时参数的类型,name参数表示信号名字,该项缺省时使用类的属性名字。
使用pyqtSignal()函数创建信号时,信号可以传递多个参数,并指定信号传递参数的类型,参数类型是标准的Python数据类型(字符串、日期、布尔类型、数字、列表、元组和字典)。
为控件创建信号
使用connect()函数可以把信号绑定到槽函数上。
使用 disconnect()函数可以解除信号与槽函数的绑定。
使用emit()函数可以发射信号。
信号与槽有三种使用方法,第一种是内置信号与槽的使用,第二种是自定义信号与槽的使用,第三种是装饰器的信号与槽的使用。第三种方法本质上是第一种方法的衍生。
内置信号与槽的使用
所谓内置信号与槽的使用,是指在发射信号时,使用窗口控件的函数,而不是自定义的函数。在信号与槽中,可以通过QObject.signal.connect将一QObject的信号连接到另一个QObject的槽函数。
from PyQt5.QtWidgets import QPushButton , QApplication , QWidget from PyQt5.QtWidgets import QMessageBox import sys app = QApplication([]) widget = QWidget() def showMsg(): QMessageBox.information(widget , "信息提示框","ok,弹出测试信息") btn = QPushButton("测试点击按钮",widget) btn.clicked.connect(showMsg) widget.show() sys.exit(app.exec_())
自定义信号与槽的使用
自定义信号与槽的使用,是指在发射信号时,不使用窗口控件的函数,而是使用自定义的函数(简单地说,就是使用 pyqtSignal类实例发射信号)。之所以要使用自定义信号与槽,是因为通过内置函数发射信号有自身的缺陷。首先,内置函数只包含一些常用的信号,有些信号的发射找不到对应的内置函数;其次,只有在特定情况下(如按钮的点击事件)才能发射这种信号;最后,内置函数传递的参数是特定的,不可以自定义。使用自定义的信号函数则没有这些缺陷。
在PyQt5编程中,自定义信号与槽的适用范围很灵活,比如因为业务需求,在程序中的某个地方需要发射一个信号,传递多种数据类型(实际上就是传递参数),然后在槽函数中接收传递过来的数据,这样就可以非常灵活地实现一些业务逻辑。
from PyQt5.QtCore import QObject , pyqtSignal # 信号对象 class QTypeSingnal(QObject): # 定义一个信号 sendmsg = pyqtSignal(object) def __init__(self): super(QTypeSingnal, self).__init__() def run(self): # 发射信号 self.sendmsg.emit("Hello DingJiaxiong!") # 槽对象 class QTypeSlot(QObject): def __init__(self): super(QTypeSlot, self).__init__() # 槽对象中的槽函数 def get(self,msg): print("QSlot get msg => " + msg) if __name__ == '__main__': # 实例化对象 send = QTypeSingnal() slot = QTypeSlot() # 1. print("--------信号绑定到槽函数----------") send.sendmsg.connect(slot.get) # 发射信号 send.run() # 2. print("--------把信号与槽函数的连接断开--------") send.sendmsg.disconnect(slot.get) send.run()
【两个参数同理】
from PyQt5.QtCore import QObject , pyqtSignal # 信号对象 class QTypeSingnal(QObject): # 定义一个信号 sendmsg = pyqtSignal(str , str) def __init__(self): super(QTypeSingnal, self).__init__() def run(self): # 发射信号 self.sendmsg.emit("Hello DingJiaxiong!","How are you?") # 槽对象 class QTypeSlot(QObject): def __init__(self): super(QTypeSlot, self).__init__() # 槽对象中的槽函数 def get(self,msg1,msg2): print("QSlot get msg => " + msg1 + msg2) if __name__ == '__main__': # 实例化对象 send = QTypeSingnal() slot = QTypeSlot() # 1. print("--------信号绑定到槽函数----------") send.sendmsg.connect(slot.get) # 发射信号 send.run() # 2. print("--------把信号与槽函数的连接断开--------") send.sendmsg.disconnect(slot.get) send.run()
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。