原文地址: http://www.cnblogs.com/chenyuming507950417/archive/2012/01/02/2310114.html
今天討論下C/C++中的回調(diào)函數(shù)沦疾。
在理解“回調(diào)函數(shù)”之前围来,首先討論下函數(shù)指針的概念硼一。
函數(shù)指針
(1)概念:指針是一個(gè)變量兢卵,是用來(lái)指向內(nèi)存地址的。一個(gè)程序運(yùn)行時(shí)价认,所有和運(yùn)行相關(guān)的物件都是需要加載到內(nèi)存中,這就決定了程序運(yùn)行時(shí)的任何物件都可以用指針來(lái)指向它。函數(shù)是存放在內(nèi)存代碼區(qū)域內(nèi)的癣疟,它們同樣有地址,因此同樣可以用指針來(lái)存取函數(shù)潮酒,把這種指向函數(shù)入口地址的指針?lè)Q為函數(shù)指針睛挚。
(2)先來(lái)看一個(gè)Hello World程序:
int main(int argc,char* argv[])
{
printf("Hello World!\n");
return 0;
}
然后,采用函數(shù)調(diào)用的形式來(lái)實(shí)現(xiàn):
void Invoke(char* s);
int main(int argc,char* argv[])
{
Invoke("Hello World!\n");
return 0;
}
void Invoke(char* s)
{
printf(s);
}
用函數(shù)指針的方式來(lái)實(shí)現(xiàn):
void Invoke(char* s);
int main()
{
void (*fp)(char* s); //聲明一個(gè)函數(shù)指針(fp)
fp=Invoke; //將Invoke函數(shù)的入口地址賦值給fp
fp("Hello World!\n"); //函數(shù)指針fp實(shí)現(xiàn)函數(shù)調(diào)用
return 0;
}
void Invoke(char* s)
{
printf(s);
}
由上知道:函數(shù)指針函數(shù)的聲明之間唯一區(qū)別就是急黎,用指針名(fp)代替了函數(shù)名Invoke扎狱,這樣這聲明了一個(gè)函數(shù)指針侧到,然后進(jìn)行賦值fp=Invoke就可以進(jìn)行函數(shù)指針的調(diào)用了。聲明函數(shù)指針時(shí)淤击,只要函數(shù)返回值類(lèi)型匠抗、參數(shù)個(gè)數(shù)、參數(shù)類(lèi)型等保持一致污抬,就可以聲明一個(gè)函數(shù)指針了汞贸。注意,函數(shù)指針必須用括號(hào)括起來(lái) void (fp)(char* s)壕吹。
實(shí)際中,為了方便删铃,通常用宏定義的方式來(lái)聲明函數(shù)指針耳贬,實(shí)現(xiàn)程序如下:
typedef void (*FP)(char* s);
void Invoke(char* s);
int main(int argc,char* argv[])
{
FP fp; //通常是用宏FP來(lái)聲明一個(gè)函數(shù)指針fp
fp=Invoke;
fp("Hello World!\n");
return 0;
}
void Invoke(char* s)
{
printf(s);
}
函數(shù)指針數(shù)組
下面用程序?qū)瘮?shù)指針數(shù)組來(lái)個(gè)大致了解:
#include <iostream>
#include <string>
using namespace std;
typedef void (*FP)(char* s);
void f1(char* s){cout<<s;}
void f2(char* s){cout<<s;}
void f3(char* s){cout<<s;}
int main(int argc,char* argv[])
{
void* a[]={f1,f2,f3}; //定義了指針數(shù)組,這里a是一個(gè)普通指針
a[0]("Hello World!\n"); //編譯錯(cuò)誤猎唁,指針數(shù)組不能用下標(biāo)的方式來(lái)調(diào)用函數(shù)
FP f[]={f1,f2,f3}; //定義一個(gè)函數(shù)指針的數(shù)組咒劲,這里的f是一個(gè)函數(shù)指針
f[0]("Hello World!\n"); //正確,函數(shù)指針的數(shù)組進(jìn)行下標(biāo)操作可以進(jìn)行函數(shù)的間接調(diào)用
return 0;
}
回調(diào)函數(shù)
(1)概念:回調(diào)函數(shù)诫隅,顧名思義腐魂,就是使用者自己定義一個(gè)函數(shù),使用者自己實(shí)現(xiàn)這個(gè)函數(shù)的程序內(nèi)容逐纬,然后把這個(gè)函數(shù)作為參數(shù)傳入別人(或系統(tǒng))的函數(shù)中蛔屹,由別人(或系統(tǒng))的函數(shù)在運(yùn)行時(shí)來(lái)調(diào)用的函數(shù)。函數(shù)是你實(shí)現(xiàn)的豁生,但由別人(或系統(tǒng))的函數(shù)在運(yùn)行時(shí)通過(guò)參數(shù)傳遞的方式調(diào)用兔毒,這就是所謂的回調(diào)函數(shù)。簡(jiǎn)單來(lái)說(shuō)甸箱,就是由別人的函數(shù)運(yùn)行期間來(lái)回調(diào)你實(shí)現(xiàn)的函數(shù)育叁。
(2)標(biāo)準(zhǔn)Hello World程序:
int main(int argc,char* argv[])
{
printf("Hello World!\n");
return 0;
}
將它修改成函數(shù)回調(diào)樣式:
//定義回調(diào)函數(shù)
void PrintfText()
{
printf("Hello World!\n");
}
//定義實(shí)現(xiàn)回調(diào)函數(shù)的"調(diào)用函數(shù)"
void CallPrintfText(void (*callfuct)())
{
callfuct();
}
//在main函數(shù)中實(shí)現(xiàn)函數(shù)回調(diào)
int main(int argc,char* argv[])
{
CallPrintfText(PrintfText);
return 0;
}
修改成帶參的回調(diào)樣式:
//定義帶參回調(diào)函數(shù)
void PrintfText(char* s)
{
printf(s);
}
//定義實(shí)現(xiàn)帶參回調(diào)函數(shù)的"調(diào)用函數(shù)"
void CallPrintfText(void (*callfuct)(char*),char* s)
{
callfuct(s);
}
//在main函數(shù)中實(shí)現(xiàn)帶參的函數(shù)回調(diào)
int main(int argc,char* argv[])
{
CallPrintfText(PrintfText,"Hello World!\n");
return 0;
}
至此,對(duì)回調(diào)函數(shù)應(yīng)該有了一個(gè)大致的了解芍殖。