PyQt訊號與槽之訊號與槽的高階用法(四)
阿新 • • 發佈:2019-01-05
裝飾器訊號與槽
所謂裝飾器訊號與槽,就是通過裝飾器的方法來定義訊號與槽函式,具體的使用方法如下
@PyQt5.QtCore.pyqtSlot(引數)
def on_傳送者物件名稱)發射訊號名稱(self,引數):
pass
這種方法有效的前提是下面的函式已經執行:
QMetaObject.connectSlotsByName(QObject)
在上面的程式碼中,‘傳送者物件名稱’就是使用setObjectName函式設定的名稱,因此自定義槽函式的命名規則也可以看做:on+使用setObjectName設定的名稱+訊號名稱
接下來是具體的例項
import sys
from PyQt5.QtWidgets import QApplication,QWidget,QHBoxLayout,QPushButton
from PyQt5 import QtCore
class CustWidget(QWidget):
def __init__(self,parent=None):
super(CustWidget, self).__init__(parent)
#建立按鈕,新增到自身視窗中
self.okButton=QPushButton('ok',self)
#使用setObjectName設定物件名稱
self.okButton.setObjectName('okButton')
#設定自身的佈局為水平佈局,並新增按鈕控制元件到其中
layout=QHBoxLayout(self)
layout.addWidget(self.okButton)
#TODo 第一種方法
#self.okButton.clicked.connect(self.okButton_clicked)
#def okButton_clicked( self ):
# print('單擊了ok按鈕')
#第二種方法
QtCore.QMetaObject.connectSlotsByName(self)
@QtCore.pyqtSlot()
def on_okButton_clicked( self ):
print('單擊了ok按鈕')
if __name__ == '__main__':
app=QApplication(sys.argv)
win=CustWidget()
win.show()
sys.exit(app.exec_())
執行程式,顯示效果如圖,單擊ok按鈕,控制檯列印預期除錯資訊
QtCore.QMetaObject.connectSlotsByName(self)
事實上,它是PyQt5中根據訊號名稱自動連線到槽函式的核心程式碼,這行程式碼用來將QObject中的子孫物件的某些物件按照其objectName連線到相應的槽函式,舉個例子說名一下
假設程式碼QtCore.QMetaObject.connectSlotsByName(self)已經執行,則下面的程式碼
@QtCore.pyqtSlot()
def on_okButton_clicked( self ):
print('單擊了ok按鈕')
會被自動識別為下面的程式碼
def __init__(self,parent=None):
self.okButton.clicked.connect(self.okButton_clicked)
def okButton_clicked(self):
print('單擊了ok按鈕')
訊號與槽的斷開與連線
有時候基於某些原因,想要臨時或永久的斷開某個訊號與槽函式的連線,這就是要說的內容
一般把connect前面加上dis就是解除繫結的意思:即disconnect
例項如下
from PyQt5.QtCore import QObject,pyqtSignal
class SignalClass(QObject):
#宣告無引數的訊號
signal1=pyqtSignal()
#宣告帶一個int型別引數的訊號
signal2=pyqtSignal(int)
def __init__(self,parent=None):
super(SignalClass, self).__init__(parent)
#將signal1訊號連線到兩個槽函式
self.signal1.connect(self.sig1Call)
self.signal1.connect(self.sig2Call)
#將signal2訊號連線到訊號1
self.signal2.connect(self.signal1)
#發射訊號
self.signal1.emit()
self.signal2.emit(1)
#斷開訊號與槽函式的關係
self.signal1.disconnect(self.sig1Call)
self.signal1.disconnect(self.sig2Call)
self.signal2.disconnect(self.signal1)
#繫結訊號與槽函式
self.signal1.connect(self.sig1Call)
self.signal2.connect(self.sig1Call)
#訊號發射
self.signal1.emit()
self.signal2.emit(1)
#輸出訊號1發射
def sig1Call( self ):
print('signal-1 emit')
#輸出訊號2發射
def sig2Call( self ):
print('signal-2 emit')
if __name__ == '__main__':
signal=SignalClass()
執行結果顯示如下