一篇文章帶你了解C/C++的回調(diào)函數(shù)
函數(shù)指針
概念
指針是一個(gè)變量,是用來指向內(nèi)存地址的。一個(gè)程序運(yùn)行時(shí),所有和運(yùn)行相關(guān)的物件都是需要加載到內(nèi)存中,這就決定了程序運(yùn)行時(shí)的任何物件都可以用指針來指向它。函數(shù)是存放在內(nèi)存代碼區(qū)域內(nèi)的,它們同樣有地址,因此同樣可以用指針來存取函數(shù),把這種指向函數(shù)入口地址的指針稱為函數(shù)指針。
先來看一個(gè)Hello World程序
int main(int argc,char* argv[]) { printf("Hello World!\n"); return 0; }
然后,采用函數(shù)調(diào)用的形式來實(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ù)指針的方式來實(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)括起來 void (*fp)(char* s)。
實(shí)際中,為了方便,通常用宏定義的方式來聲明函數(shù)指針,實(shí)現(xiàn)程序如下:
typedef void (*FP)(char* s); void Invoke(char* s); int main(int argc,char* argv[]) { FP fp; //通常是用宏FP來聲明一個(gè)函數(shù)指針fp fp=Invoke; fp("Hello World!\n"); return 0; } void Invoke(char* s) { printf(s); }
函數(shù)指針數(shù)組
下面用程序?qū)瘮?shù)指針數(shù)組來個(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)的方式來調(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í)來調(diào)用的函數(shù)。函數(shù)是你實(shí)現(xiàn)的,但由別人(或系統(tǒng))的函數(shù)在運(yùn)行時(shí)通過參數(shù)傳遞的方式調(diào)用,這就是所謂的回調(diào)函數(shù)。簡單來說,就是由別人的函數(shù)運(yùn)行期間來回調(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è)通過函數(shù)指針調(diào)用的函數(shù)。如果你把函數(shù)的指針(地址)作為參數(shù)傳遞給另一個(gè)函數(shù),當(dāng)這個(gè)指針被用為調(diào)用它所指向的函數(shù)時(shí),我們就說這是回調(diào)函數(shù)?;卣{(diào)函數(shù)不是由該函數(shù)的實(shí)現(xiàn)方直接調(diào)用,而是在特定的事件或條件發(fā)生時(shí)由另外的一方調(diào)用的,用于對該事件或條件進(jìn)行響應(yīng)。
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
C語言入門篇--四大常量(字面,const修飾,宏,枚舉)及標(biāo)識(shí)符
本篇文章是c語言基礎(chǔ)篇,主要講述一下常量,常量即不可被直接修改的量(const修飾的常變量可間接修改,后續(xù)文章會(huì)繼續(xù)說明)請大家持續(xù)關(guān)注腳本之家2021-08-08c++多線程之死鎖的發(fā)生的情況解析(包含兩個(gè)歸納,6個(gè)示例)
這篇文章主要介紹了c++多線程之死鎖的發(fā)生的情況解析(包含兩個(gè)歸納,6個(gè)示例),需要的朋友可以參考下2018-01-01C++定時(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-09C語言數(shù)據(jù)結(jié)構(gòu)系列之樹的概念結(jié)構(gòu)和常見表示方法
本章將正式開啟數(shù)據(jù)結(jié)構(gòu)中?“樹”?部分的講解,本章將介紹樹的概念和結(jié)構(gòu),以及樹的表示方法,感興趣的朋友進(jìn)來看看吧2022-02-02VS2019添加引用出錯(cuò):對COM組件的調(diào)用返回了錯(cuò)誤HRESULT E_FAIL(未能完成操作未指定的錯(cuò)誤)
這篇文章主要介紹了VS2019添加引用出錯(cuò):對COM組件的調(diào)用返回了錯(cuò)誤HRESULT E_FAIL(未能完成操作。未指定的錯(cuò)誤),需要的朋友可以參考下2020-07-07C++實(shí)現(xiàn)LeetCode(129.求根到葉節(jié)點(diǎn)數(shù)字之和)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(129.求根到葉節(jié)點(diǎn)數(shù)字之和),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07