http://wuzsh860916.blog.163.com/blog/static/55836620072259454119/一 通常的函數調用
一個(gè)通常的函數調用的例子:
//自行包含頭文件
void MyFun(int x); //此處的申明也可寫(xiě)成:void MyFun( int );
int main(int argc, char* argv[])
{
MyFun(10); //這里是調用MyFun(10);函數
return 0;
}
void MyFun(int x) //這里定義一個(gè)MyFun函數
{
printf(“%d\n”,x);
}
這個(gè)MyFun函數是一個(gè)無(wú)返回值的函數,它并不完成什么事情。這種調用函數的格式你應該是很熟悉的吧!看主函數中調用MyFun函數的書(shū)寫(xiě)格式:
MyFun(10);
我們一開(kāi)始只是從功能上或者說(shuō)從數學(xué)意義上理解MyFun這個(gè)函數,知道MyFun函數名代表的是一個(gè)功能(或是說(shuō)一段代碼)。
直到——
學(xué)習到函數指針概念時(shí)。我才不得不在思考:函數名到底又是什么東西呢?
(不要以為這是沒(méi)有什么意義的事噢!呵呵,繼續往下看你就知道了。)
二 函數指針變量的申明
就象某一數據變量的內存地址可以存儲在相應的指針變量中一樣,函數的首地址也以存儲在某個(gè)函數指針變量里的。這樣,我就可以通過(guò)這個(gè)函數指針變量來(lái)調用所指向的函數了。
在C系列語(yǔ)言中,任何一個(gè)變量,總是要先申明,之后才能使用的。那么,函數指針變量也應該要先申明吧?那又是如何來(lái)申明呢?以上面的例子為例,我來(lái)申明一個(gè)可以指向MyFun函數的函數指針變量FunP。下面就是申明FunP變量的方法:
void (*FunP)(int) ; //也可寫(xiě)成void (*FunP)(int x);
你看,整個(gè)函數指針變量的申明格式如同函數MyFun的申明處一樣,只不過(guò)——我們把MyFun改成(*FunP)而已,這樣就有了一個(gè)能指向MyFun函數的指針FunP了。(當然,這個(gè)FunP指針變量也可以指向所有其它具有相同參數及返回值的函數了。)
三 通過(guò)函數指針變量調用函數
有了FunP指針變量后,我們就可以對它賦值指向MyFun,然后通過(guò)FunP來(lái)調用MyFun函數了??次胰绾瓮ㄟ^(guò)FunP指針變量來(lái)調用MyFun函數的:
//自行包含頭文件
void MyFun(int x); //這個(gè)申明也可寫(xiě)成:void MyFun( int );
void (*FunP)(int ); //也可申明成void(*FunP)(int x),但習慣上一般不這樣。
int main(int argc, char* argv[])
{
MyFun(10); //這是直接調用MyFun函數
FunP=&MyFun; //將MyFun函數的地址賦給FunP變量
(*FunP)(20); //這是通過(guò)函數指針變量FunP來(lái)調用MyFun函數的。
}
void MyFun(int x) //這里定義一個(gè)MyFun函數
{
printf(“%d\n”,x);
}
請看黑體字部分的代碼及注釋。
運行看看。嗯,不錯,程序運行得很好。
哦,我的感覺(jué)是:MyFun與FunP的類(lèi)型關(guān)系類(lèi)似于int 與int *的關(guān)系。函數MyFun好像是一個(gè)如int的變量(或常量),而FunP則像一個(gè)如int *一樣的指針變量。
int i,*pi;
pi=&i; //與FunP=&MyFun比較。
(你的感覺(jué)呢?)
呵呵,其實(shí)不然——
四 調用函數的其它書(shū)寫(xiě)格式
函數指針也可如下使用,來(lái)完成同樣的事情:
//自行包含頭文件
void MyFun(int x);
void (*FunP)(int ); //申明一個(gè)用以指向同樣參數,返回值函數的指針變量。
int main(int argc, char* argv[])
{
MyFun(10); //這里是調用MyFun(10);函數
FunP=MyFun; //將MyFun函數的地址賦給FunP變量
FunP(20); //這是通過(guò)函數指針變量來(lái)調用MyFun函數的。
return 0;
}
void MyFun(int x) //這里定義一個(gè)MyFun函數
{
printf(“%d\n”,x);
}
我改了黑體字部分(請自行與之前的代碼比較一下)。
運行試試,??!一樣地成功。
咦?
FunP=MyFun;
可以這樣將MyFun值同賦值給FunP,難道MyFun與FunP是同一數據類(lèi)型(即如同的int 與int的關(guān)系),而不是如同int 與int*的關(guān)系了?(有沒(méi)有一點(diǎn)點(diǎn)的糊涂了?)
看來(lái)與之前的代碼有點(diǎn)矛盾了,是吧!所以我說(shuō)嘛!
請容許我暫不給你解釋?zhuān)^續看以下幾種情況(這些可都是可以正確運行的代碼喲?。?div style="height:15px;">
代碼之三:
int main(int argc, char* argv[])
{
MyFun(10); //這里是調用MyFun(10);函數
FunP=&MyFun; //將MyFun函數的地址賦給FunP變量
FunP(20); //這是通過(guò)函數指針變量來(lái)調用MyFun函數的。
return 0;
}
代碼之四:
int main(int argc, char* argv[])
{
MyFun(10); //這里是調用MyFun(10);函數
FunP=MyFun; //將MyFun函數的地址賦給FunP變量
(*FunP)(20); //這是通過(guò)函數指針變量來(lái)調用MyFun函數的。
return 0;
}
真的是可以這樣的噢!
(哇!真是要暈倒了?。?div style="height:15px;">
1. 其實(shí),MyFun的函數名與FunP函數指針都是一樣的,即都是函數指針。MyFun函數名是一個(gè)函數指針常量,而FunP是一個(gè)函數數指針變量,這是它們的關(guān)系。
2. 但函數名調用如果都得如(*MyFun)(10);這樣,那書(shū)寫(xiě)與讀起來(lái)都是不方便和不習慣的。所以C語(yǔ)言的設計者們才會(huì )設計成又可允許MyFun(10);這種形式地調用(這樣方便多了并與數學(xué)中的函數形式一樣,不是嗎?)。
3. 為統一起見(jiàn),FunP函數指針變量也可以FunP(10)的形式來(lái)調用。
4. 賦值時(shí),即可FunP=&MyFun形式,也可FunP=MyFun。
根據注釋?zhuān)瑧摬浑y看懂吧?。m然你可能很少這樣定義使用,但以后學(xué)習Win32編程時(shí)會(huì )經(jīng)常見(jiàn)到的。)