python學(xué)習(xí)之hook鉤子的原理和使用
什么是鉤子
之前有轉(zhuǎn)一篇關(guān)于回調(diào)函數(shù)的文章
鉤子函數(shù)、注冊(cè)函數(shù)、回調(diào)函數(shù),他們的概念其實(shí)是一樣的。
鉤子函數(shù),顧名思義,就是把我們自己實(shí)現(xiàn)的hook函數(shù)在某一時(shí)刻掛接到目標(biāo)掛載點(diǎn)上。
1. hook函數(shù),就是我們自己實(shí)現(xiàn)的函數(shù),函數(shù)類型與掛載點(diǎn)匹配(返回值,參數(shù)列表)
2. 掛接,也就是hook或者叫注冊(cè)(register),使得hook函數(shù)對(duì)目標(biāo)可用
3. 目標(biāo)掛載點(diǎn),也就是掛我們hook函數(shù)的地方(我們想在這個(gè)目標(biāo)點(diǎn)實(shí)現(xiàn)我們自己的功能)
先看一張圖:
hook的概念在windows的消息響應(yīng)機(jī)制里面體現(xiàn)的尤為明顯??赡芪覀兇蠹矣袑?xiě)過(guò)windows桌面相關(guān)的程序(像MFC),里面有各種消息監(jiān)聽(tīng)響應(yīng)機(jī)制。比如,要監(jiān)聽(tīng)鼠標(biāo)左鍵是否按下這個(gè)事件,我們要去實(shí)現(xiàn)一個(gè)onLeftKeyDown()之類的方法,該方法可以稱為鉤子函數(shù)。同時(shí),我們還要去注冊(cè)鉤子函數(shù),MFC中是通過(guò)一組宏來(lái)實(shí)現(xiàn)的。這樣當(dāng)鼠標(biāo)左鍵按下后,就能調(diào)到我們定義的方法了。
為什么需要鉤子
大家思考一下上面這個(gè)例子,左鍵按下方法具體的邏輯是由框架自身去實(shí)現(xiàn),還是由我們用戶(調(diào)用者)去實(shí)現(xiàn)呢?顯然應(yīng)該由我們自己去實(shí)現(xiàn)。要提供通用的框架能力,框架自身去實(shí)現(xiàn)該方法功能,是沒(méi)有意義的,所以框架給提供一個(gè)掛載的point,把具體邏輯的實(shí)現(xiàn)交給用戶就好了,靈活可用。
鉤子使用
hook是一個(gè)編程機(jī)制,與語(yǔ)言無(wú)關(guān)。這里給個(gè)python的簡(jiǎn)單例子,幫助大家理解:
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()
代碼運(yùn)行結(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
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
基于pytorch中的Sequential用法說(shuō)明
這篇文章主要介紹了基于pytorch中的Sequential用法說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-06-06Python強(qiáng)制子類重寫(xiě)父類的兩種方法實(shí)現(xiàn)
在Python中,有時(shí)候我們希望某個(gè)類能夠被子類繼承并重寫(xiě)其中的某些方法,本文主要介紹了Python強(qiáng)制子類重寫(xiě)父類的兩種方法實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-02-02Python文件讀寫(xiě)處理日常任務(wù)終極工具實(shí)例
Python文件的讀寫(xiě)操作時(shí),有很多需要考慮的細(xì)節(jié),這包括文件打開(kāi)方式、讀取和寫(xiě)入數(shù)據(jù)的方法、異常處理等,在本文中,將深入探討Python中的文件操作,旨在提供全面的指南,幫你充分了解Python文件的讀寫(xiě)2023-11-11Python的數(shù)據(jù)結(jié)構(gòu)與算法的隊(duì)列詳解(3)
這篇文章主要為大家詳細(xì)介紹了Python的隊(duì)列,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-03-03python中的 Matplotlib 繪制多子圖時(shí)的重疊問(wèn)題及解決方案
當(dāng)使用 Matplotlib 繪制多個(gè)子圖(subplots)時(shí),如果標(biāo)簽或標(biāo)題文字交叉或重疊,遇到這樣的問(wèn)題如何處理呢,下面小編給大家介紹了python中的 Matplotlib 繪制多子圖時(shí)的重疊問(wèn)題及解決方案,需要的朋友可以參考下2024-06-06Python 使用 consul 做服務(wù)發(fā)現(xiàn)示例詳解
這篇文章主要介紹了Python 使用 consul 做服務(wù)發(fā)現(xiàn)示例詳解,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03Python實(shí)現(xiàn)在線暴力破解郵箱賬號(hào)密碼功能示例【測(cè)試可用】
這篇文章主要介紹了Python實(shí)現(xiàn)在線暴力破解郵箱賬號(hào)密碼功能,結(jié)合完整實(shí)例形式分析了Python讀取txt字典文件針對(duì)郵箱的相關(guān)驗(yàn)證破解操作技巧,需要的朋友可以參考下2017-09-09