1. 程式人生 > >C/C++之回調函數

C/C++之回調函數

logs pri 方便 函數回調 yum www 宏定義 -a ()

C/C++之回調函數

  在理解“回調函數”之前,首先討論下函數指針的概念。

函數指針

(1)概念:指針是一個變量,是用來指向內存地址的。一個程序運行時,所有和運行相關的物件都是需要加載到內存中,這就決定了程序運行時的任何物件都可以用指針來指向它。函數是存放在內存代碼區域內的,它們同樣有地址,因此同樣可以用指針來存取函數,把這種指向函數入口地址的指針稱為函數指針。

(2)先來看一個Hello World程序:

1 int main(int argc,char* argv[])
2 {
3     printf("Hello World!\n");
4     return 0;
5 }

  然後,采用函數調用的形式來實現:

 1 void Invoke(char* s);
 2 
 3 int main(int argc,char* argv[])
 4 {
 5     Invoke("Hello World!\n");
 6     return 0;
 7 }
 8 
 9 void Invoke(char* s)
10 {
11     printf(s);
12 }

  用函數指針的方式來實現:

 1 void Invoke(char* s);
 2 
 3 int main()
 4 {
 5     void (*fp)(char* s);    //聲明一個函數指針(fp)        
 6     fp=Invoke;              //
將Invoke函數的入口地址賦值給fp 7 fp("Hello World!\n"); //函數指針fp實現函數調用 8 return 0; 9 } 10 11 void Invoke(char* s) 12 { 13 printf(s); 14 }

  由上知道:函數指針函數的聲明之間唯一區別就是,用指針名(*fp)代替了函數名Invoke,這樣這聲明了一個函數指針,然後進行賦值fp=Invoke就可以進行函數指針的調用了。聲明函數指針時,只要函數返回值類型、參數個數、參數類型等保持一致,就可以聲明一個函數指針了。註意,函數指針必須用括號括起來 void (*fp)(char* s)。

  實際中,為了方便,通常用宏定義的方式來聲明函數指針,實現程序如下:

 1 typedef void (*FP)(char* s);
 2 void Invoke(char* s);
 3 
 4 int main(int argc,char* argv[])
 5 {
 6     FP fp;      //通常是用宏FP來聲明一個函數指針fp
 7     fp=Invoke;
 8     fp("Hello World!\n");
 9     return 0;
10 }
11 
12 void Invoke(char* s)
13 {
14     printf(s);
15 }

函數指針數組

下面用程序對函數指針數組來個大致了解:

 1 #include <iostream>
 2 #include <string>
 3 using namespace std;
 4 
 5 typedef void (*FP)(char* s);
 6 void f1(char* s){cout<<s;}
 7 void f2(char* s){cout<<s;}
 8 void f3(char* s){cout<<s;}
 9 
10 int main(int argc,char* argv[])
11 {
12     void* a[]={f1,f2,f3};   //定義了指針數組,這裏a是一個普通指針
13     a[0]("Hello World!\n"); //編譯錯誤,指針數組不能用下標的方式來調用函數
14 
15     FP f[]={f1,f2,f3};      //定義一個函數指針的數組,這裏的f是一個函數指針
16     f[0]("Hello World!\n"); //正確,函數指針的數組進行下標操作可以進行函數的間接調用
17     
18     return 0;
19 }

回調函數

(1)概念:回調函數,顧名思義,就是使用者自己定義一個函數,使用者自己實現這個函數的程序內容,然後把這個函數作為參數傳入別人(或系統)的函數中,由別人(或系統)的函數在運行時來調用的函數。函數是你實現的,但由別人(或系統)的函數在運行時通過參數傳遞的方式調用,這就是所謂的回調函數。簡單來說,就是由別人的函數運行期間來回調你實現的函數。

(2)標準Hello World程序:

1 int main(int argc,char* argv[])
2 {
3     printf("Hello World!\n");
4     return 0;
5 }

   將它修改成函數回調樣式:

 1 //定義回調函數
 2 void PrintfText() 
 3 {
 4     printf("Hello World!\n");
 5 }
 6 
 7 //定義實現回調函數的"調用函數"
 8 void CallPrintfText(void (*callfuct)())
 9 {
10     callfuct();
11 }
12 
13 //在main函數中實現函數回調
14 int main(int argc,char* argv[])
15 {
16     CallPrintfText(PrintfText);
17     return 0;
18 }

  修改成帶參的回調樣式:

 1 //定義帶參回調函數
 2 void PrintfText(char* s) 
 3 {
 4     printf(s);
 5 }
 6 
 7 //定義實現帶參回調函數的"調用函數"
 8 void CallPrintfText(void (*callfuct)(char*),char* s)
 9 {
10     callfuct(s);
11 }
12 
13 //在main函數中實現帶參的函數回調
14 int main(int argc,char* argv[])
15 {
16     CallPrintfText(PrintfText,"Hello World!\n");
17     return 0;
18 }

  至此,對回調函數應該有了一個大致的了解。

如有不對的地方,非常歡迎給予指導!

——【感謝】資料來源於http://www.cnblogs.com/chenyuming507950417/archive/2012/01/02/2310114.html

C/C++之回調函數