前言
在 Qt 中可以使用信號和槽機制很方便地實現部件之間的通信,考慮下面這樣的場景:
我想要點擊任意一個專輯卡並通知主界面跳轉到專輯界面,那么一種實現方式如上圖所示:點擊任意一個藍色方框所示的專輯卡,發出 switchToAlbumIntetrfaceSig
給父級部件專輯卡視圖,因為專輯卡視圖有許多個分組,比如上圖中為 aiko
分組,可能還有 柳井愛子
分組,那么這些視圖都應該將 switchToAlbumInterfaceSig
轉發給父級窗口我的音樂界面,我的音樂界面再轉發給主界面,從而實現界面跳轉。
可以看到上面這種做法很麻煩,專輯卡上擁有 switchToAlbumInterfaceSig
屬性就算了,還要連累父級專輯卡視圖以及祖父級我的音樂界面也擁有這個屬性才能實現信號的轉發。有沒有一種方式可以省掉中間的轉發過程,從而一步到位通知主界面呢?這就需要使用下面所介紹的全局事件總線思想。
全局事件總線
Vue 中的全局事件總線
在 vue 中要實現任意組件間通信,可以在 Vue.prototype
上添加一個全局事件總線 $bus
屬性,當組件 A 想要給組件 B 發送一些數據時,只需要在 A 中 this.$bus.$emit(事件名,數據)
發送數據,在 B 中 this.$bus.$on(事件名,回調)
就能通過總線收到數據,而無需借助其他組件的轉發。將事件名視為信號,回調視為槽函數,那么這個過程和 Qt 的信號和槽機制神似。
Qt 中的全局事件總線
仿照上述過程,我們來定義一個全局事件總線類,並使用單例模式保證只能實例化出一個對象:
# coding:utf-8
from PyQt5.QtCore import QObject, pyqtSignal
class SignalBus(QObject):
""" 全局事件總線 """
switchToAlbumInterfaceSig = pyqtSignal(str)
def __new__(cls, *args, **kwargs):
if not hasattr(cls, '_instance'):
cls._instance = super(SignalBus, cls).__new__(cls, *args, **kwargs)
return cls._instance
bus = SignalBus()
回到最初的那個例子,現在我們只需導入 bus
對象,點擊 aikoの詩。
專輯卡時 bus.switchToAlbumInterfaceSig.emit('aiko - aikoの詩。')
來發送切換到專輯界面的信號,然后在主界面中 bus.switchToAlbumInterfaceSig.connect(self.switchToAlbumInterface)
即可,這樣就省去了信號的轉發流程,代碼會簡潔許多。(界面的實現代碼在 Groove,據說把倉庫從 public 變為 private 之后 star 會被清空,別問我是怎么知道的 😊)以上~~