什么是回調(diào)函數(shù)?
簡而言之,回調(diào)函數(shù)就是一個通過函數(shù)指針調(diào)用的函數(shù)。如果你把函數(shù)的指針(地址)作為參數(shù)傳遞給另一個函數(shù),當這個指針被用為調(diào)用它所指向的函數(shù)時,我們就說這是回調(diào)函數(shù)。
為什么要使用回調(diào)函數(shù)?
因為可以把調(diào)用者與被調(diào)用者分開。調(diào)用者不關(guān)心誰是被調(diào)用者,所有它需知道的,只是存在一個具有某種特定原型、某些限制條件(如返回值為int)的被調(diào)用函數(shù)。
如果想知道回調(diào)函數(shù)在實際中有什么作用,先假設(shè)有這樣一種情況,我們要編寫一個庫,它提供了某些排序算法的實現(xiàn),如冒泡排序、快速排序、shell排序、shake排序等等,但為使庫更加通用,不想在函數(shù)中嵌入排序邏輯,而讓使用者來實現(xiàn)相應(yīng)的邏輯;或者,想讓庫可用于多種數(shù)據(jù)類型(int、float、string),此時,該怎么辦呢?可以使用函數(shù)指針,并進行回調(diào)。
回調(diào)可用于通知機制,例如,有時要在程序中設(shè)置一個計時器,每到一定時間,程序會得到相應(yīng)的通知,但通知機制的實現(xiàn)者對我們的程序一無所知。而此時,就需有一個特定原型的函數(shù)指針,用這個指針來進行回調(diào),來通知我們的程序事件已經(jīng)發(fā)生。實際上,SetTimer() API使用了一個回調(diào)函數(shù)來通知計時器,而且,萬一沒有提供回調(diào)函數(shù),它還會把一個消息發(fā)往程序的消息隊列。
另一個使用回調(diào)機制的API函數(shù)是EnumWindow(),它枚舉屏幕上所有的頂層窗口,為每個窗口調(diào)用一個程序提供的函數(shù),并傳遞窗口的處理程序。如果被調(diào)用者返回一個值,就繼續(xù)進行迭代,否則,退出。EnumWindow()并不關(guān)心被調(diào)用者在何處,也不關(guān)心被調(diào)用者用它傳遞的處理程序做了什么,它只關(guān)心返回值,因為基于返回值,它將繼續(xù)執(zhí)行或退出。
不管怎么說,回調(diào)函數(shù)是繼續(xù)自C語言的,因而,在C++中,應(yīng)只在與C代碼建立接口,或與已有的回調(diào)接口打交道時,才使用回調(diào)函數(shù)。除了上述情況,在C++中應(yīng)使用虛擬方法或函數(shù)符(functor),而不是回調(diào)函數(shù)。
下面是自己寫的一個簡單的回調(diào)函數(shù),相比其他的那些復(fù)雜的代碼,這個更容易理解: #include<stdio.h>
#include<stdlib.h>
void perfect(int n)
{
int i=1;
int count=0;
for(i=1;i<n;i++)
{
if(0==n%i)
{
count+=i;
}
}
if(count==n)
printf("%d是完數(shù)\n",n);
else printf("%d不是完數(shù)\n",n);
}
void myCallback(void (*perfect)(int ),int n)
{
perfect(n);
} int main()
{
int n;
printf("請輸入一個正整數(shù)\n");
scanf("%d",&n); myCallback(perfect,n);
return 0;
}
|