1、 什麼是回調函數?
首先什麼是「回調」呢?
我的理解是:把一段可執行的代碼像參數傳遞那樣傳給其他代碼,而這段代碼會在某個時刻被調用執行,這就叫做回調。
如果代碼立即被執行就稱為同步回調,如果過後再執行,則稱之為異步回調。
回調函數就是一個通過函數指針調用的函數。如果你把函數的指針(地址)作為參數傳遞給另一個函數,當這個指針被用來調用其所指向的函數時,我們就說這是回調函數。
回調函數不是由該函數的實現方直接調用,而是在特定的事件或條件發生時由另外的一方調用的,用於對該事件或條件進行響應。
2、 為什麼要用回調函數?
因為可以把調用者與被調用者分開,所以調用者不關心誰是被調用者。它只需知道存在一個具有特定原型和限制條件的被調用函數。
簡而言之,回調函數就是允許用戶把需要調用的方法的指針作為參數傳遞給一個函數,以便該函數在處理相似事件的時候可以靈活的使用不同的方法。
int Callback() ///
{
// TODO
return 0;
}
int main() ///
{
// TODO
Library(Callback); ///
// TODO
return 0;
}
回調似乎只是函數間的調用,和普通函數調用沒啥區別。
但仔細看,可以發現兩者之間的一個關鍵的不同:在回調中,主程序把回調函數像參數一樣傳入庫函數。
這樣一來,只要我們改變傳進庫函數的參數,就可以實現不同的功能,這樣有沒有覺得很靈活?並且當庫函數很複雜或者不可見的時候利用回調函數就顯得十分優秀
3 、怎麼使用回調函數?
int Callback_1(int a) ///
{
printf("Hello, this is Callback_1: a = %d ", a);
return 0;
}
int Callback_2(int b) ///
{
printf("Hello, this is Callback_2: b = %d ", b);
return 0;
}
int Callback_3(int c) ///
{
printf("Hello, this is Callback_3: c = %d ", c);
return 0;
}
int Handle(int x, int (*Callback)(int)) ///
{
Callback(x);
}
int main()
{
Handle(4, Callback_1);
Handle(5, Callback_2);
Handle(6, Callback_3);
return 0;
}
如上述代碼:可以看到,Handle()函數裡面的參數是一個指針,在main()函數裡調用Handle()函數的時候,給它傳入了函數Callback_1()/Callback_2()/Callback_3()的函數名,這時候的函數名就是對應函數的指針,也就是說,回調函數其實就是函數指針的一種用法。
4 、回調函數實例(很有用)
一個GPRS模塊聯網的小項目,使用過的同學大概知道2G、4G、NB等模塊要想實現無線聯網功能都需要經歷模塊上電初始化、註冊網絡、查詢網絡信息質量、連接伺服器等步驟,這裡的的例子就是,利用一個狀態機函數(根據不同狀態依次調用不同實現方法的函數),通過回調函數的方式依次調用不同的函數,實現模塊聯網功能,如下:
/********* 工作狀態處理 *********/
typedef struct
{
uint8_t mStatus;
uint8_t (* Funtion)(void); //函數指針的形式
} M26_WorkStatus_TypeDef; //M26的工作狀態集合調用函數
/**********************************************
** >M26工作狀態集合函數
***********************************************/
M26_WorkStatus_TypeDef M26_WorkStatus_Tab[] =
{
, //模塊關機
, //模塊開機
, //管腳初始化
, /AT指令配置
, //連接調度中心
, //等待調度中心回復
, //連接前置機
, //等待前置機回復
, //正常工作
, //等待信號回復
, //獲取信號值
, //模塊重啟
}
/**********************************************
** >M26模塊工作狀態機,依次調用裡面的12個函數
***********************************************/
uint8_t M26_WorkStatus_Call(uint8_t Start)
{
uint8_t i = 0;
for(i = 0; i
{
if(Start == M26_WorkStatus_Tab[i].mStatus)
{
return M26_WorkStatus_Tab[i].Funtion();
}
}
return 0;
}