1. 程式人生 > >PyQt訊號與槽之訊號與槽的高階用法(四)

PyQt訊號與槽之訊號與槽的高階用法(四)

裝飾器訊號與槽

所謂裝飾器訊號與槽,就是通過裝飾器的方法來定義訊號與槽函式,具體的使用方法如下

@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()

執行結果顯示如下
這裡寫圖片描述