python學(xué)習(xí)之hook鉤子的原理和使用
什么是鉤子
之前有轉(zhuǎn)一篇關(guān)于回調(diào)函數(shù)的文章
鉤子函數(shù)、注冊函數(shù)、回調(diào)函數(shù),他們的概念其實是一樣的。
鉤子函數(shù),顧名思義,就是把我們自己實現(xiàn)的hook函數(shù)在某一時刻掛接到目標(biāo)掛載點上。
1. hook函數(shù),就是我們自己實現(xiàn)的函數(shù),函數(shù)類型與掛載點匹配(返回值,參數(shù)列表)
2. 掛接,也就是hook或者叫注冊(register),使得hook函數(shù)對目標(biāo)可用
3. 目標(biāo)掛載點,也就是掛我們hook函數(shù)的地方(我們想在這個目標(biāo)點實現(xiàn)我們自己的功能)
先看一張圖:
hook的概念在windows的消息響應(yīng)機制里面體現(xiàn)的尤為明顯。可能我們大家有寫過windows桌面相關(guān)的程序(像MFC),里面有各種消息監(jiān)聽響應(yīng)機制。比如,要監(jiān)聽鼠標(biāo)左鍵是否按下這個事件,我們要去實現(xiàn)一個onLeftKeyDown()之類的方法,該方法可以稱為鉤子函數(shù)。同時,我們還要去注冊鉤子函數(shù),MFC中是通過一組宏來實現(xiàn)的。這樣當(dāng)鼠標(biāo)左鍵按下后,就能調(diào)到我們定義的方法了。
為什么需要鉤子
大家思考一下上面這個例子,左鍵按下方法具體的邏輯是由框架自身去實現(xiàn),還是由我們用戶(調(diào)用者)去實現(xiàn)呢?顯然應(yīng)該由我們自己去實現(xiàn)。要提供通用的框架能力,框架自身去實現(xiàn)該方法功能,是沒有意義的,所以框架給提供一個掛載的point,把具體邏輯的實現(xiàn)交給用戶就好了,靈活可用。
鉤子使用
hook是一個編程機制,與語言無關(guān)。這里給個python的簡單例子,幫助大家理解:
import time class LazyPerson(object): def __init__(self, name): self.name = name self.watch_tv_func = None self.have_dinner_func = None def get_up(self): print("%s get up at:%s" % (self.name, time.time())) def go_to_sleep(self): print("%s go to sleep at:%s" % (self.name, time.time())) def register_tv_hook(self, watch_tv_func): self.watch_tv_func = watch_tv_func def register_dinner_hook(self, have_dinner_func): self.have_dinner_func = have_dinner_func def enjoy_a_lazy_day(self): # get up self.get_up() time.sleep(3) # watch tv # check the watch_tv_func(hooked or unhooked) # hooked if self.watch_tv_func is not None: self.watch_tv_func(self.name) # unhooked else: print("no tv to watch") time.sleep(3) # have dinner # check the have_dinner_func(hooked or unhooked) # hooked if self.have_dinner_func is not None: self.have_dinner_func(self.name) # unhooked else: print("nothing to eat at dinner") time.sleep(3) self.go_to_sleep() def watch_daydayup(name): print("%s : The program ---day day up--- is funny!!!" % name) def watch_happyfamily(name): print("%s : The program ---happy family--- is boring!!!" % name) def eat_meat(name): print("%s : The meat is nice!!!" % name) def eat_hamburger(name): print("%s : The hamburger is not so bad!!!" % name) if __name__ == "__main__": lazy_tom = LazyPerson("Tom") lazy_jerry = LazyPerson("Jerry") # register hook lazy_tom.register_tv_hook(watch_daydayup) lazy_tom.register_dinner_hook(eat_meat) lazy_jerry.register_tv_hook(watch_happyfamily) lazy_jerry.register_dinner_hook(eat_hamburger) # enjoy a day lazy_tom.enjoy_a_lazy_day() lazy_jerry.enjoy_a_lazy_day()
代碼運行結(jié)果:
Tom get up at:1509246940.32 Tom : The program ---day day up--- is funny!!! Tom : The meat is nice!!! Tom go to sleep at:1509246949.34 Jerry get up at:1509246949.34 Jerry : The program ---happy family--- is boring!!! Jerry : The hamburger is not so bad!!! Jerry go to sleep at:1509246958.37
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python的數(shù)據(jù)結(jié)構(gòu)與算法的隊列詳解(3)
這篇文章主要為大家詳細(xì)介紹了Python的隊列,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-03-03python中的 Matplotlib 繪制多子圖時的重疊問題及解決方案
當(dāng)使用 Matplotlib 繪制多個子圖(subplots)時,如果標(biāo)簽或標(biāo)題文字交叉或重疊,遇到這樣的問題如何處理呢,下面小編給大家介紹了python中的 Matplotlib 繪制多子圖時的重疊問題及解決方案,需要的朋友可以參考下2024-06-06Python 使用 consul 做服務(wù)發(fā)現(xiàn)示例詳解
這篇文章主要介紹了Python 使用 consul 做服務(wù)發(fā)現(xiàn)示例詳解,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-03-03Python實現(xiàn)在線暴力破解郵箱賬號密碼功能示例【測試可用】
這篇文章主要介紹了Python實現(xiàn)在線暴力破解郵箱賬號密碼功能,結(jié)合完整實例形式分析了Python讀取txt字典文件針對郵箱的相關(guān)驗證破解操作技巧,需要的朋友可以參考下2017-09-09