C/C++ 回調函數詳解


【1】函數指針


【1.1】概念


指針是一個變量,是用來指向內存地址的。一個程序運行時,所有和運行相關的物件都是需要加載到內存中,這
就決定了程序運行時的任何物件都可以用指針來指向它。函數是存放在內存代碼區域內的,它們同樣有地址,因此同
樣可以用指針來存取函數,把這種指向函數入口地址的指針稱為函數指針。

【1.2】Hello World程序


int _tmain(int argc, _TCHAR* argv[])
{
printf("Hello World!\n");
return 0;
}

用函數調用方式實現:

void Invoke(char* s)
{
printf(s);
}

int _tmain(int argc, _TCHAR* argv[])
{
Invoke("Hello World!\n");
return 0;
}

用函數指針方式實現:

typedef void (* PINVOKE)(char *s);

void Invoke(char* s)
{
printf(s);
}

int _tmain(int argc, _TCHAR* argv[])
{
PINVOKE fp = Invoke;
fp("Hello World!\n");
return 0;
}

由上知道:函數指針與函數的聲明之間唯一區別就是,用指針名(*fp)代替了函數名Invoke,這樣聲明了一
個函數指針,然后進行賦值fp=Invoke就可以進行函數指針的調用了。聲明函數指針時,只要函數返回值類型、參
數個數、參數類型等保持一致,就可以聲明一個函數指針了。注意,函數指針必須用括號括起來 void (*fp)
(char* s)。

【2】回調函數


【2.1】概念


回調函數,顧名思義,就是使用者自己定義一個函數,使用者自己實現這個函數的程序內容,然后把這個函數
作為參數傳入別人(或系統)的函數中,由別人(或系統)的函數在運行時來調用的函數。函數是你實現的,但由別人(或系統)的函數在運行時通過參數傳遞的方式調用,這就是所謂的回調函數。簡單來說,就是由別人的函數運行
期間來回調你實現的函數

【2.2】示例


typedef void (*CALLFUN)( char* );
void CallPrintfText(CALLFUN fp, char* s)
{
fp(s);
}

void PrintfText(char* s)
{
printf(s);
}

int _tmain(int argc, _TCHAR* argv[])
{
CallPrintfText(PrintfText, "Hello World!\n");
return 0;
}

【3】類成員函數作為回調函數


回調函數是基於C的Windows SDK的技術,不是針對C++的,程序員可以將一個C函數直接作為回調函
數, 但是如果試圖直接使用C++的成員函數作為回調函數將發生錯誤。

普通的C++成員函數都隱含了一個傳遞函數作為參數,亦即“this”指針,C++通過傳遞一個指向自身的指針給
其成員函數從而實現程序函數可以訪問C++的數據成員。  

所以使用類的成員函數作為回調函數的解決方法有兩種:不使用成員函數(使用友元操作符friend的C函數訪問類的數據成員);使用靜態成員函數
示例如下(該示例解決靜態成員函數的只能訪問靜態的限制)

#include "stdafx.h"
#include <iostream>
using namespace std;

class CPrintString
{
public:
CPrintString(){};
~CPrintString(){};

public:
void PrintfText(const char* s){ printf(s); };
static void StaticPrintText(void* ptClass, const char* s);
};
void CPrintString::StaticPrintText(void* ptClass, const char* s)
{
CPrintString* ptThis = static_cast<CPrintString*>(ptClass);
if (NULL == ptClass)
{
return;
}

ptThis->PrintfText(s);
}

typedef void (*PPRINTTEXT)(void* ptClass, const char* s);
void CallPrintText(void* ptClass, const char *s, PPRINTTEXT fp)
{
fp(ptClass, s);
}

int _tmain(int argc, _TCHAR* argv[])
{
CPrintString obj;
CallPrintText((void*)&obj, "Hello World!\n", CPrintString::StaticPrintText);
return 0;
}

參考文章:
http://www.cnblogs.com/chenyuming507950417/archive/2012/01/02/2310114.html

注意!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系我们删除。



 
粤ICP备14056181号  © 2014-2021 ITdaan.com