一個函數(shù)指針的理解:
有一段程序存儲在起始地址為 0的一段內(nèi)存上,如果我們想要調(diào)用這段程序,請問該如何去做?
答案是 (*(void (*)( ) )0)( )。
首先,最基本的函數(shù)聲明: void function (paramList);
最基本的函數(shù)調(diào)用: function(paramList);
鑒于問題中的函數(shù)沒有參數(shù),函數(shù)調(diào)用可簡化為 function();
根據(jù)問題描述,可以知道 0是這個函數(shù)的入口地址,也就是說,0是一個函數(shù)的指針。
使用函數(shù)指針的函數(shù)聲明形式是:void (*pFunction)(),相應(yīng)的調(diào)用形式是: (*pFunction)(),
則問題中的函數(shù)調(diào)用可以寫作:(*0)( )。
大家知道,函數(shù)指針變量不能是一個常數(shù),因此上式中的 0必須要被轉(zhuǎn)化為函數(shù)指針。
我們先來研究一下,對于使用函數(shù)指針的函數(shù):
比如 void (*pFunction)( ),函數(shù)指針變量的原型是什么?
這個問題很簡單,pFunction函數(shù)指針原型是( void (*)( ) ),即去掉變量名,
清晰起見,整個加上()號。
所以將 0強制轉(zhuǎn)換為一個返回值為void,參數(shù)為空的函數(shù)指針如下:( void (*)( ) )。
OK,結(jié)合2)和3)的分析,結(jié)果出來了,那就是:(*(void (*)( ) )0)( ) 。
答案分析:從頭到尾理解答案
(void (*)( )) ,是一個返回值為void,參數(shù)為空的函數(shù)指針原型。
(void (*)( ))0,把0轉(zhuǎn)變成一個返回值為void,參數(shù)為空的函數(shù)指針,指針指向的地址為0.
*(void (*)( ))0,前面加上*表示整個是一個返回值為void的函數(shù)的名字
(*(void (*)( ))0)( ),這當(dāng)然就是一個函數(shù)了。
我們可以使用 typedef清晰聲明如下:
typedef void (*pFun)();
這樣定義之后,pFun就是一個返回類型為void無參數(shù)的函數(shù)指針變量了。
這樣函數(shù)變?yōu)?(*(pFun)0)();
----
在調(diào)用動態(tài)庫時,習(xí)慣用typedef重新定義動態(tài)庫函數(shù)中的函數(shù)地址(函數(shù)指針),
如在動態(tài)庫(test.dll)中有如下函數(shù):
int DoCase(int, long);
則,在調(diào)用動態(tài)庫是有兩種方法:
1. 先聲明一個與動態(tài)庫中類型一致的指針函數(shù)變量:
int (*DOCASE)(int ,long); //用于指向動態(tài)庫中的DoCase函數(shù)地址
HINSTANCE gLibMyDLL = NULL;
gLibMyDLL = LoadLibrary("test.dll");
if(gLibMyDLL != NULL)
{
DOCASE = (int(*)(int,long))GetProcAddress(gLibMyDLL, "DoCase");
}
int s = DOCASE(1,1000);
2.用typedef定義一個指針函數(shù):
typedef (*DOCASE)(int ,long);
HINSTANCE gLibMyDLL = NULL;
DOCASE _docase;
gLibMyDLL = LoadLibrary("test.dll");
if(gLibMyDLL != NULL)
{
_docase = (DOCASE)GetProcAddress(gLibMyDll, "DoCase");
}
int s=_docase(1,1000);
----------------
在C++類中使用函數(shù)指針。
//typedef 返回類型(類名::*新類型)(參數(shù)表)
class CA
{
public:
char lcFun(int a){ return; }
};
CA ca;
typedef char (CA::*PTRFUN)(int);
PTRFUN pFun;
void main()
{
pFun = CA::lcFun;
ca.(*pFun)(2);
}
指針的定義與使用都加上了“類限制”或“對象”,用來指明指針指向的函數(shù)是哪個類的,
這里的類對象也可以是使用new得到的。
如:
CA *pca = new CA;
pca->(*pFun)(2);
delete pca;
而且這個類對象指針可以是類內(nèi)部成員變量,你甚至可以使用this指針。
如:類CA有成員變量PTRFUN m_pfun;
void CA::lcFun2()
{
(this->*m_pFun)(2);
}
一句話,使用類成員函數(shù)指針必須有“->*”或“.*”的調(diào)用。
--------------------
void test(void* );
void tt()
{
printf("kao,沒玩過這種\n");
}
int main(int argc, char* argv[])
{
typedef void(*Fun)();
Fun mytest;
mytest = tt;
test((void*)mytest);
return 0;
}
void test(void * p)
{
(*(void(*)())p)();
}