1. 程式人生 > 實用技巧 >python實現事件驅動模型

python實現事件驅動模型

轉:https://www.jianshu.com/p/a605fab0ab11

# encoding: UTF-8
# 系統模組
from Queue import Queue, Empty
from threading import *
########################################################################
class EventManager:
    #----------------------------------------------------------------------
    def __init__
(self): """初始化事件管理器""" # 事件物件列表 self.__eventQueue = Queue() # 事件管理器開關 self.__active = False # 事件處理執行緒 self.__thread = Thread(target = self.__Run) # 這裡的__handlers是一個字典,用來儲存對應的事件的響應函式 # 其中每個鍵對應的值是一個列表,列表中儲存了對該事件監聽的響應函式,一對多 self.__handlers
= {} #---------------------------------------------------------------------- def __Run(self): """引擎執行""" while self.__active == True: try: # 獲取事件的阻塞時間設為1秒 event = self.__eventQueue.get(block = True, timeout = 1) self.
__EventProcess(event) except Empty: pass #---------------------------------------------------------------------- def __EventProcess(self, event): """處理事件""" # 檢查是否存在對該事件進行監聽的處理函式 if event.type_ in self.__handlers: # 若存在,則按順序將事件傳遞給處理函式執行 for handler in self.__handlers[event.type_]: handler(event) #---------------------------------------------------------------------- def Start(self): """啟動""" # 將事件管理器設為啟動 self.__active = True # 啟動事件處理執行緒 self.__thread.start() #---------------------------------------------------------------------- def Stop(self): """停止""" # 將事件管理器設為停止 self.__active = False # 等待事件處理執行緒退出 self.__thread.join() #---------------------------------------------------------------------- def AddEventListener(self, type_, handler): """繫結事件和監聽器處理函式""" # 嘗試獲取該事件型別對應的處理函式列表,若無則建立 try: handlerList = self.__handlers[type_] except KeyError: handlerList = [] self.__handlers[type_] = handlerList # 若要註冊的處理器不在該事件的處理器列表中,則註冊該事件 if handler not in handlerList: handlerList.append(handler) #---------------------------------------------------------------------- def RemoveEventListener(self, type_, handler): """移除監聽器的處理函式""" #讀者自己試著實現 #---------------------------------------------------------------------- def SendEvent(self, event): """傳送事件,向事件佇列中存入事件""" self.__eventQueue.put(event) ######################################################################## """事件物件""" class Event: def __init__(self, type_=None): self.type_ = type_ # 事件型別 self.dict = {} # 字典用於儲存具體的事件資料

測試程式碼

#-------------------------------------------------------------------
# encoding: UTF-8
import sys
from datetime import datetime
from threading import *
from EventManager import *

#事件名稱  新文章
EVENT_ARTICAL = "Event_Artical"

#事件源 公眾號
class PublicAccounts:
    def __init__(self,eventManager):
        self.__eventManager = eventManager

    def WriteNewArtical(self):
        #事件物件,寫了新文章
        event = Event(type_=EVENT_ARTICAL)
        event.dict["artical"] = u'如何寫出更優雅的程式碼\n'
        #傳送事件
        self.__eventManager.SendEvent(event)
        print u'公眾號傳送新文章\n'

#監聽器 訂閱者
class Listener:
    def __init__(self,username):
        self.__username = username

    #監聽器的處理函式 讀文章
    def ReadArtical(self,event):
        print(u'%s 收到新文章' % self.__username)
        print(u'正在閱讀新文章內容:%s'  % event.dict["artical"])

"""測試函式"""
#--------------------------------------------------------------------
def test():
    listner1 = Listener("thinkroom") #訂閱者1
    listner2 = Listener("steve")#訂閱者2

    eventManager = EventManager()
    
    #繫結事件和監聽器響應函式(新文章)
    eventManager.AddEventListener(EVENT_ARTICAL, listner1.ReadArtical)
    eventManager.AddEventListener(EVENT_ARTICAL, listner2.ReadArtical)
    eventManager.Start()

    publicAcc = PublicAccounts(eventManager)
    timer = Timer(2, publicAcc.WriteNewArtical)
    timer.start()
    
if __name__ == '__main__':
    test()