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