回調(diào)函數(shù)以及鉤子函數(shù)的概念
鉤子實(shí)際上是一個(gè)處理消息的程序段,通過系統(tǒng)調(diào)用,把它掛入系統(tǒng)。每當(dāng)特定的消息發(fā)出,在沒有到達(dá)目的窗口前,鉤子程序就先捕獲該消息,亦即鉤子函數(shù)先得到控制權(quán)。這時(shí)鉤子函數(shù)即可以加工處理(改變)該消息,也可以不作處理而繼續(xù)傳遞該消息,還可以強(qiáng)制結(jié)束消息的傳遞。
對(duì)每種類型的鉤子由系統(tǒng)來維護(hù)一個(gè)鉤子鏈,最近安裝的鉤子放在鏈的開始,而***安裝的鉤子放在***,也就是后加入的先獲得控制權(quán)。要實(shí)現(xiàn)Win32的系統(tǒng)鉤子,必須調(diào)用SDK中的API函數(shù)SetWindowsHookEx來安裝這個(gè)鉤子函數(shù),這個(gè)函數(shù)的原型是HHOOK SetWindowsHookEx(int idHook,HOOKPROC lpfn,HINSTANCE hMod,DWORD dwThreadId);其中,***個(gè)參數(shù)是鉤子的類型;第二個(gè)參數(shù)是鉤子函數(shù)的地址;第三個(gè)參數(shù)是包含鉤子函數(shù)的模塊句柄;第四個(gè)參數(shù)指定監(jiān)視的線程。
如果指定確定的線程,即為線程專用鉤子;如果指定為空,即為全局鉤子。其中,全局鉤子函數(shù)必須包含在DLL(動(dòng)態(tài)鏈接庫)中,而線程專用鉤子還可以包含在可執(zhí)行文件中。得到控制權(quán)的鉤子函數(shù)在完成對(duì)消息的處理后,如果想要該消息繼續(xù)傳遞,那么它必須調(diào)用另外一個(gè)SDK中的API函數(shù)CallNextHookEx來傳遞它。鉤子函數(shù)也可以通過直接返回TRUE來丟棄該消息,并阻止該消息的傳遞。
下面這篇文章寫回調(diào)函數(shù)的概念還是比較清晰的,回調(diào)函數(shù)就是自己寫的一個(gè)函數(shù),但是不能被顯式的調(diào)用,而是把該函數(shù)的地址作為一個(gè)別的函數(shù)參數(shù)來引用,這樣用來處理當(dāng)一些事件發(fā)生時(shí)可以調(diào)用這個(gè)自己定義的回調(diào)函數(shù),完成一些處理。
回調(diào)函數(shù)大多只是自己定義一個(gè)名字而已,函數(shù)體大多是系統(tǒng)定義好的,它有一個(gè)結(jié)構(gòu),一般一個(gè)代回調(diào)函數(shù)的的函數(shù)都有一個(gè)參數(shù)是接你的回調(diào)名的,它把一些值傳進(jìn)回調(diào)函數(shù)(函數(shù)體包括參數(shù)是它預(yù)定好的,不能自己寫,除非全部函數(shù)都是你寫的),然后回調(diào)函數(shù)接受值,相應(yīng)操作后將值返回到原函數(shù)體(它的父親函數(shù)),最終讓原函數(shù)返回一個(gè)值
我們經(jīng)常在 C++ 設(shè)計(jì)時(shí)通過使用回調(diào)函數(shù)可以使有些應(yīng)用(如定時(shí)器事件回調(diào)處理、用回調(diào)函數(shù)記錄某操作進(jìn)度等)變得非常方便和符合邏輯,那么它的內(nèi)在機(jī)制如何呢,怎么定義呢 ? 它和其它函數(shù)(比如鉤子函數(shù))有何不同呢?這里結(jié)合自己的使用經(jīng)歷做一個(gè)簡(jiǎn)單的介紹。
使用回調(diào)函數(shù)實(shí)際上就是在調(diào)用某個(gè)函數(shù)(通常是 API 函數(shù))時(shí),將自己的一個(gè)函數(shù)(這個(gè)函數(shù)為回調(diào)函數(shù))的地址作為參數(shù)傳遞給那個(gè)函數(shù)。而那個(gè)函數(shù)在需要的時(shí)候,利用傳遞的地址調(diào)用回調(diào)函數(shù),這時(shí)你可以利用這個(gè)機(jī)會(huì)在回調(diào)函數(shù)中處理消息或完成一定的操作。至于如何定義回調(diào)函數(shù),跟具體使用的 API 函數(shù)有關(guān),一般在幫助中有說明回調(diào)函數(shù)的參數(shù)和返回值等。 C++ 中一般要求在回調(diào)函數(shù)前加 CALLBACK (相當(dāng)于 FAR PASCAL ),這主要是說明該函數(shù)的調(diào)用方式。
至于鉤子函數(shù),只是回調(diào)函數(shù)的一個(gè)特例。習(xí)慣上把與 SetWindowsHookEx 函數(shù)一起使用的回調(diào)函數(shù)稱為鉤子函數(shù)。也有人把利用 VirtualQueryEx 安裝的函數(shù)稱為鉤子函數(shù),不過這種叫法不太流行。
也可以這樣,更容易理解:回調(diào)函數(shù)就好像是一個(gè)中斷處理函數(shù),系統(tǒng)在符合你設(shè)定的條件時(shí)自動(dòng)調(diào)用。為此,你需要做三件事:
1. 聲明;
2. 定義;
3. 設(shè)置觸發(fā)條件,就是在你的函數(shù)中把你的回調(diào)函數(shù)名稱轉(zhuǎn)化為地址作為一個(gè)參數(shù),以便于系統(tǒng)調(diào)用。
聲明和定義時(shí)應(yīng)注意:回調(diào)函數(shù)由系統(tǒng)調(diào)用,所以可以認(rèn)為它屬于 WINDOWS 系統(tǒng),不要把它當(dāng)作你的某個(gè)類的成員函數(shù)
回調(diào)函數(shù) 是一個(gè)程序員不能顯式調(diào)用的函數(shù);通過將回調(diào)函數(shù) 的地址傳給調(diào)用者從而實(shí)現(xiàn)調(diào)用?;卣{(diào)函數(shù) 使用是必要的,在我們想通過一個(gè)統(tǒng)一接口實(shí)現(xiàn)不同的內(nèi)容,這時(shí)用回掉函數(shù)非常合適。
比如,我們?yōu)閹讉€(gè)不同的設(shè)備分別寫了不同的顯示函數(shù):
void TVshow(); void ComputerShow(); void NoteBookShow()...等等。
這是我們想用一個(gè)統(tǒng)一的顯示函數(shù),我們這時(shí)就可以用回掉函數(shù)了。void show(void (*ptr)()); 使用時(shí)根據(jù)所傳入的參數(shù)不同而調(diào)用不同的回調(diào)函數(shù) 。
不同的編程語言可能有不同的語法,下面舉一個(gè)c語言中回調(diào)函數(shù) 的例子,其中一個(gè)回調(diào)函數(shù) 不帶參數(shù),另一個(gè)回調(diào)函數(shù) 帶參數(shù)。
例子1:
- //Test.c
- #include <stdlib.h>
- #include <stdio.h>
- int Test1()
- {
- int i;
- for (i=0; i<30; i++)
- {
- printf("The %d th charactor is: %c\n", i, (char)('a' + i%26));
- }
- return 0;
- }
- int Test2(int num)
- {
- int i;
- for (i=0; i<num; i++)
- {
- printf("The %d th charactor is: %c\n", i, (char)('a' + i%26));
- }
- return 0;
- }
- void Caller1(void (*ptr)())//指向函數(shù)的指針作函數(shù)參數(shù)
- {
- (*ptr)();
- }
- void Caller2(int n, int (*ptr)())//指向函數(shù)的指針作函數(shù)參數(shù),這里***個(gè)參數(shù)是為指向函數(shù)的指針服務(wù)的,
- { //不能寫成void Caller2(int (*ptr)(int n)),這樣的定義語法錯(cuò)誤。
- (*ptr)(n);
- return;
- }
- int main()
- {
- printf("************************\n");
- Caller1(Test1); //相當(dāng)于調(diào)用Test2();
- printf("&&&&&&************************\n");
- Caller2(30, Test2); //相當(dāng)于調(diào)用Test2(30);
- return 0;
- }
以上通過將回調(diào)函數(shù) 的地址傳給調(diào)用者從而實(shí)現(xiàn)調(diào)用,但是需要注意的是帶參回調(diào)函數(shù) 的用法。要實(shí)現(xiàn)回調(diào),必須首先定義函數(shù)指針。函數(shù)指針的定義這里稍微提一下。比如:
int (*ptr)(); 這里ptr是一個(gè)函數(shù)指針,其中(*ptr)的括號(hào)不能省略,因?yàn)槔ㄌ?hào)的優(yōu)先級(jí)高于星號(hào),那樣就成了一個(gè)返回類型為整型的函數(shù)聲明了。
【編輯推薦】