對于回調(diào)函數(shù)的編寫始終是寫特殊處理功能程序時(shí)用到的技巧之一。先介紹一下回調(diào)的使用基本方法與原理。
1、在這里設(shè):回調(diào)函數(shù)為A()(這是最簡單的情況,不帶參數(shù),但我們應(yīng)用的實(shí)際情況常常很會(huì)復(fù)雜),使用回調(diào)函數(shù)的操作函數(shù)為B(), 但B函數(shù)是需要參數(shù)的,這個(gè)參數(shù)就是指向函數(shù)A的地址變量,這個(gè)變量一般就是函數(shù)指針。使用方法為:
int A(char *p); // 回調(diào)函數(shù)
typedef int(*CallBack)(char *p) ; // 聲明CallBack 類型的函數(shù)指針
CallBack myCallBack ; // 聲明函數(shù)指針變量
myCallBack = A; // 得到了函數(shù)A的地址 B函數(shù)一般會(huì)寫為 B(CallBack lpCall,char * P,........); // 此處省略了p后的參數(shù)形式 。
所以回調(diào)機(jī)制可解為,函數(shù)B要完成一定功能,但他自己是無法實(shí)現(xiàn)全部功能的。 需要借助于函數(shù)A來完成,也就是回調(diào)函數(shù)。B的實(shí)現(xiàn)為:
B(CallBack lpCall,char *pProvide)
{
........... // B 的自己實(shí)現(xiàn)功能語句
lpCall(PpProvide); // 借助回調(diào)完成的功能 ,也就是A函數(shù)來處理的。
........... // B 的自己實(shí)現(xiàn)功能語句
}
// -------------- 使用例子 -------------
char *p = "hello!";
CallBack myCallBack ;
myCallBack = A ;
B(A, p); 以上就是回調(diào)的基本應(yīng)用,本文所說的變身,其實(shí)是利用傳入不同的函數(shù)地址,實(shí)現(xiàn)調(diào)用者類與回調(diào)函數(shù)所在類的不同轉(zhuǎn)換。
1、問題描述
CUploadFile 類完成數(shù)據(jù)上傳,與相應(yīng)的界面進(jìn)度顯示。
主要函數(shù)Send(...) 和回調(diào)函數(shù) GetCurState() ;
class CUploadFile : public CDialog
{
......
int Send(LPCTSTR lpServerIP, LPCTSTR lpServerPort, LPCTSTR UploadFilePath) ;
static int GetCurState(int nCurDone, int nInAll, void * pParam) ;
......
}
int CUploadFile ::Send(LPCTSTR lpServerIP, LPCTSTR lpServerPort, LPCTSTR UploadFilePath)
{
... // 導(dǎo)出傳輸數(shù)據(jù)的函數(shù)
int ret = Upload( (LPSTR)(LPCTSTR)m_strData,
GetCurState, // 在這個(gè)回調(diào)函數(shù)中處理界面
this, // CUploadFile 的自身指針 ,也就是pParam 所接受的參數(shù)
(LPSTR)(LPCTSTR)UploadFilePath,
"",
"",
);
}
int CUploadFile ::GetCurState(int nCurData, int nInAll, void * pParam)
{
.........
UploadFile *pThis = (UploadFile *)pParam; // nCurData 當(dāng)前以傳出的數(shù)據(jù)量
// nInAll 總的數(shù)據(jù)量
// 有了pThis可以對界面進(jìn)行各種操作了。
.............
} 但大家仔細(xì)觀察就可以發(fā)現(xiàn),這個(gè)類把數(shù)據(jù)傳送和界面顯示聚和到了一起,不容易得到復(fù)用。而且在復(fù)用過程中需要改動(dòng)較多的地方 。
請大家記住現(xiàn)在的回調(diào)函數(shù)傳入的類本身的靜態(tài)成員函數(shù)。
現(xiàn)在我們把數(shù)據(jù)的傳送和界面的顯示分離。回調(diào)則要傳入的是界面處理類的靜態(tài)函數(shù)。
界面處理類 CShowGUI,數(shù)據(jù)上傳類 CUploadData
class CUploadData
{
......
typedef int(*SetUploadCaller)(int nCurData, int nInAll, void * pParam);
int UploadFile(LPCTSTR lpFileNamePath,LPVOID lparam,SetUploadCaller Caller );
// 接受外界出入的參數(shù),主要是回調(diào)函數(shù)的地址通過參數(shù)Caller,
int Send(LPCTSTR lpServerIP, LPCTSTR lpServerPort, LPCTSTR UploadFilePath) ;
...... // 注意此時(shí)不在需要GetCurState 函數(shù)了 。
}
class CShowGUI: public CDialog
{
.......
typedef int(*SetUploadCaller)(int nCurData, int nInAll, void * pParam);
void SetCallBack(LPCTSTR strPath);
static int GetCurState(int nCurData, int nInAll, void * pParam) ;
CUploadData m_Uploa
d ; // 數(shù)據(jù)上傳類是界面顯示類的一個(gè)成員變量。
.......
}
void CShowGUI :: SetCallBack(LPCTSTR strPath)
{
CUploadData myUploadData ;
SetUploadCaller myCaller; // 聲明一個(gè)函數(shù)指針變量
myCaller = CurState ; // 取得界面處理函數(shù)的地址
myUploadData .UploadFile(strPath,this,myCaller); // 界面處理類的函數(shù)傳入,實(shí)現(xiàn)了數(shù)據(jù)傳入與界面處理的分離 .
} 通過上面的演示做到了界面與數(shù)據(jù)的分離,回調(diào)函數(shù)分別扮演了不同角色,所以隨著處理問題的不同應(yīng)靈活應(yīng)用,但同樣因?yàn)樘幚頂?shù)據(jù)類不知道界面處理類或外部調(diào)用類的類型,而更無法靈活地處理界面的不同顯示方式。這方面還希望喜歡鉆研技術(shù)的朋友繼續(xù)研究。