一篇文章帶你了解C/C++的回調(diào)函數(shù)
函數(shù)指針
概念
指針是一個(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ù)指針。
先來(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ù)返回值類型、參數(shù)個(gè)數(shù)、參數(shù)類型等保持一致,就可以聲明一個(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ù)
概念
回調(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ù)。
標(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;
}總結(jié)
回調(diào)函數(shù)就是一個(gè)通過(guò)函數(shù)指針調(diào)用的函數(shù)。如果你把函數(shù)的指針(地址)作為參數(shù)傳遞給另一個(gè)函數(shù),當(dāng)這個(gè)指針被用為調(diào)用它所指向的函數(shù)時(shí),我們就說(shuō)這是回調(diào)函數(shù)?;卣{(diào)函數(shù)不是由該函數(shù)的實(shí)現(xiàn)方直接調(diào)用,而是在特定的事件或條件發(fā)生時(shí)由另外的一方調(diào)用的,用于對(duì)該事件或條件進(jìn)行響應(yīng)。
本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
C語(yǔ)言入門篇--四大常量(字面,const修飾,宏,枚舉)及標(biāo)識(shí)符
本篇文章是c語(yǔ)言基礎(chǔ)篇,主要講述一下常量,常量即不可被直接修改的量(const修飾的常變量可間接修改,后續(xù)文章會(huì)繼續(xù)說(shuō)明)請(qǐng)大家持續(xù)關(guān)注腳本之家2021-08-08
c++多線程之死鎖的發(fā)生的情況解析(包含兩個(gè)歸納,6個(gè)示例)
這篇文章主要介紹了c++多線程之死鎖的發(fā)生的情況解析(包含兩個(gè)歸納,6個(gè)示例),需要的朋友可以參考下2018-01-01
C++定時(shí)器實(shí)現(xiàn)和時(shí)間輪介紹
這篇文章主要介紹了C++定時(shí)器實(shí)現(xiàn)和時(shí)間輪介紹,定時(shí)器可以由很多種數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn),比如最小堆、紅黑樹、跳表、甚至數(shù)組都可以,其本質(zhì)都是拿到最小時(shí)間的任務(wù),然后取出該任務(wù)并執(zhí)行,更多相關(guān)內(nèi)容介紹,需要的小伙伴可以參考一下2022-09-09
C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)系列之樹的概念結(jié)構(gòu)和常見(jiàn)表示方法
本章將正式開啟數(shù)據(jù)結(jié)構(gòu)中?“樹”?部分的講解,本章將介紹樹的概念和結(jié)構(gòu),以及樹的表示方法,感興趣的朋友進(jìn)來(lái)看看吧2022-02-02
VS2019添加引用出錯(cuò):對(duì)COM組件的調(diào)用返回了錯(cuò)誤HRESULT E_FAIL(未能完成操作未指定的錯(cuò)誤)
這篇文章主要介紹了VS2019添加引用出錯(cuò):對(duì)COM組件的調(diào)用返回了錯(cuò)誤HRESULT E_FAIL(未能完成操作。未指定的錯(cuò)誤),需要的朋友可以參考下2020-07-07
C++實(shí)現(xiàn)LeetCode(129.求根到葉節(jié)點(diǎn)數(shù)字之和)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(129.求根到葉節(jié)點(diǎn)數(shù)字之和),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07

