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

