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