背景
之前寫過一些關於Native DLL封裝的文章,連結如下:
Windows Mobile和Wince(Windows Embedded CE)下如何封裝Native DLL提供給.NET Compact Framework進行調用
Windows Mobile和Wince(Windows Embedded CE)下封裝Native DLL進一步探討
如何在Windows Mobile下使用Native C++動態載入DLL
緣由
有時候這個封裝的DLL需要為Client提供非同步通知,需要使用到回呼函數。如果不是純C的DLL下,進行非同步通知,我一般使用Oberver模式的方式來通知Client。但是在封裝純C的Native DLL,就不能使用該方法了。關於Oberver模式,請參考 我的實用設計模式之Observer模式
解決方案
下面以一個LCD touch screen的溫度和觸摸回呼函數的封裝作為例子講述如何封裝回呼函數。
定義
#ifdef LCD_EXPORTS
#define LCD_API extern "C" __declspec(dllexport)
#else
#define LCD_API extern "C" __declspec(dllimport)
#endif
/**
* Function signature for callback event handlers.
*/
typedef void (__stdcall *LcdTemperatureHandler) (double degree);
typedef void (__stdcall *LcdTouchSensorHandler) ();
LCD_API void LcdTemperatureRegisterHandler( LcdTemperatureHandler handler );
LCD_API void LcdTouchSensorRegisterHandler( LcdTouchSensorHandler handler );
LCD_API void LcdTemperatureUnregisterHandler();
LCD_API void LcdTouchSensorUnregisterHandler();
分別定義溫度和觸摸的回呼函數的函數指標LcdTemperatureHandler和LcdTouchSensorHandler。LcdTemperatureHandler回調的時候同時傳遞溫度的值(degree)。LcdTemperatureRegisterHandler和LcdTouchSensorRegisterHandler為註冊函數,參數為以LcdTemperatureHandler和LcdTouchSensorHandler為原先的全域函數或者靜態成員函數。
實現
LcdTemperatureHandler gLcdTemperatureHandler = NULL;
LcdTouchSensorHandler gLcdTouchSensorHandler = NULL;
void LcdTemperatureRegisterHandler( LcdTemperatureHandler handler )
{
gLcdTemperatureHandler = handler;
}
void LcdTouchSensorRegisterHandler( LcdTouchSensorHandler handler )
{
gLcdTouchSensorHandler = handler;
}
void LcdTemperatureUnregisterHandler()
{
gLcdTemperatureHandler = NULL;
}
void LcdTouchSensorUnregisterHandler()
{
gLcdTouchSensorHandler = NULL;
}
實現的時候需要一個全域變數或者靜態成員變數儲存函數指標。
void LcdTouched()
{
if( gLcdTouchSensorHandler != NULL)
{
gLcdTouchSensorHandler();
}
}
void LcdTemperatureChanged()
{
double temperature = 30.2;
if( gLcdTemperatureHandler != NULL)
{
gLcdTemperatureHandler(temperature);
}
}
需要的時候調用這些函數指標來回調登入的函數。
Client
Client使用的時候需要定義與LcdTemperatureHandler和LcdTouchSensorHandler一樣的函數。
void OnLcdTouched ()
{
printf("OnLcdouched\n");
}
void OnTemperatureChanged (double degree)
{
printf("OnTemperatureChanged degree=%d\n", degree);
}
把這些函數作為參數傳遞到註冊函數。
LcdTouchSensorRegisterHandler(OnLcdTouched);
LcdTemperatureRegisterHandler(OnTemperatureChanged);