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)的尤為明顯??赡芪覀兇蠹矣袑戇^(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)文章
python高手之路python處理excel文件(方法匯總)
用python來(lái)自動(dòng)生成excel數(shù)據(jù)文件。python處理excel文件主要是第三方模塊庫(kù)xlrd、xlwt、xluntils和pyExcelerator,除此之外,python處理excel還可以用win32com和openpyxl模塊2016-01-01
anaconda虛擬環(huán)境默認(rèn)路徑的更改圖文教程
在Anaconda中如果沒(méi)有指定路徑,虛擬環(huán)境會(huì)默認(rèn)安裝在anaconda所安裝的目錄下,這篇文章主要給大家介紹了關(guān)于anaconda虛擬環(huán)境默認(rèn)路徑更改的相關(guān)資料,需要的朋友可以參考下2023-10-10
Python使用正則實(shí)現(xiàn)計(jì)算字符串算式
這篇文章主要介紹了Python使用正則實(shí)現(xiàn)計(jì)算字符串算式的方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-12-12
Matplotlib animation模塊實(shí)現(xiàn)動(dòng)態(tài)圖
這篇文章主要介紹了Matplotlib animation模塊實(shí)現(xiàn)動(dòng)態(tài)圖,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02
詳解python實(shí)現(xiàn)讀取郵件數(shù)據(jù)并下載附件的實(shí)例
這篇文章主要介紹了詳解python讀取郵件數(shù)據(jù)并下載附件的實(shí)例的相關(guān)資料,這里提供實(shí)現(xiàn)實(shí)例,幫助大家學(xué)習(xí)理解這部分內(nèi)容,需要的朋友可以參考下2017-08-08
解決Python3 被PHP程序調(diào)用執(zhí)行返回亂碼的問(wèn)題
今天小編就為大家分享一篇解決Python3 被PHP程序調(diào)用執(zhí)行返回亂碼的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-02-02
使用Python實(shí)現(xiàn)將多表分批次從數(shù)據(jù)庫(kù)導(dǎo)出到Excel
這篇文章主要介紹了使用Python實(shí)現(xiàn)將多表分批次從數(shù)據(jù)庫(kù)導(dǎo)出到Excel,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-05-05

