學習了數(shù)組之后,我們知道數(shù)組是在內存中申請一塊內存空間;數(shù)組名代表內存塊的首地址,通過數(shù)組名可以訪問內存塊中的數(shù)據。
創(chuàng)新互聯(lián)服務項目包括清原網站建設、清原網站制作、清原網頁制作以及清原網絡營銷策劃等。多年來,我們專注于互聯(lián)網行業(yè),利用自身積累的技術優(yōu)勢、行業(yè)經驗、深度合作伙伴關系等,向廣大中小型企業(yè)、政府機構等提供互聯(lián)網行業(yè)的解決方案,清原網站推廣取得了明顯的社會效益與經濟效益。目前,我們服務的客戶以成都為中心已經輻射到清原省份的部分城市,未來相信會繼續(xù)擴大服務區(qū)域并繼續(xù)獲得客戶的支持與信任!
那么,對于函數(shù),它也是存放在內存塊中的一段數(shù)據。例如下面的函數(shù):
void func( int a )
{
printf( "in func, a = %d " , a );
}
此時,定義了一個函數(shù)名是func的函數(shù)??梢匀缦抡{用該函數(shù):
func(100);
此時,就進入了func函數(shù)的函數(shù)體中執(zhí)行。可以看到, 函數(shù)名如同數(shù)組名一樣,代表函數(shù)所在內存塊的首地址 。通過數(shù)組名可以訪問數(shù)組在內存塊中申請的內存,同理,通過函數(shù)名,可以訪問函數(shù)在內存中存放的數(shù)據。
所以,函數(shù)名就代表了該函數(shù)在內存塊中存放的首地址。那么,函數(shù)名是表示一個地址,就可以把這個地址值存放在某一個指針變量中,然后,通過指針變量訪問函數(shù)名指向的函數(shù)。
在C語言中,提供了函數(shù)指針變量,可以存放函數(shù)名表示的地址。函數(shù)指針變量的定義格式如下:
返回數(shù)據類型 (*函數(shù)指針變量名)(形參列表)
對比函數(shù)的定義如下:
返回數(shù)據類型 函數(shù)名(形參列表)
可以看到,函數(shù)指針變量的定義,與函數(shù)的定義格式基本一樣,唯一的區(qū)別是把“函數(shù)名”轉換為“*(函數(shù)指針變量名)”;總結如下:
(1) 使用指針降級運算符*來定義,表示這個是一個指針。
(2) 指針降級運算符*不可以靠近返回數(shù)據類型,例如“返回數(shù)據類*”就表示函數(shù)的返回類型是一個指針。那么,為了讓指針降級運算符*能夠修飾函數(shù)指針變量,就用小括號()把指針降級運算符*與函數(shù)指針變量名包含起來。
定義了函數(shù)指針變量之后,可以把函數(shù)名賦給函數(shù)指針變量。因為,函數(shù)名就表示函數(shù)在內存塊中的首地址,所以,可以直接把一個地址賦值給函數(shù)指針變量。格式如下:
函數(shù)指針變量 = 函數(shù)名;
最終,可以通過函數(shù)指針變量調用函數(shù),調用的格式與通過函數(shù)名調用完全一樣,通過函數(shù)指針變量調用函數(shù),有如下形式:
方法1:函數(shù)指針變量(實參列表);
方法2:(*函數(shù)指針變量名)(實參列表);
很多情況下,我們更傾向于使用第一種形式,因為,它的使用方式更接近于通過函數(shù)名調用函數(shù)。
下面根據程序測試例子來看看怎么樣應用函數(shù)指針變量。
深入學習,可以交個朋友,工人人人號:韋凱峰linux編程學堂
程序運行結果如下:
深入學習,可以交個朋友,工人人人號:韋凱峰linux編程學堂
可以看到,我們定義了func函數(shù)和函數(shù)指針變量pfunc,然后,把函數(shù)名func設置給函數(shù)指針變量pfunc,最終,通過函數(shù)指針變量pfunc調用函數(shù)。
因為函數(shù)指針變量存放的就是函數(shù)名表示的地址,所以,函數(shù)指針變量與函數(shù)名一樣,可以直接通過函數(shù)指針變量調用函數(shù)。
注意:我們在學習指針的時候,可以把一個int類型的變量地址賦值給int類型的指針;但是,不可以把int類型變量的地址,賦值給double類型的指針。這就是變量數(shù)據類型不一致的問題。
同樣的道理,定義函數(shù)的時候,函數(shù)的返回數(shù)據類型和形參列表都不一樣,所以,函數(shù)指針變量能夠接收的函數(shù)名,它們定義的 函數(shù)返回數(shù)據類型和形參列表必須一致 ,此時,就如同變量與指針變量類型一致時,才可以把變量的地址賦值給指針變量一樣。
如下是一個測試例子:
深入學習,可以交個朋友,工人人人號:韋凱峰linux編程學堂
程序編譯結果如下:
深入學習,可以交個朋友,工人人人號:韋凱峰linux編程學堂
可以看到,我們把func函數(shù)的形參列表修改為double,但是,函數(shù)指針變量pfunc定義的形參列表為int類型,此時,函數(shù)和函數(shù)指針變量的定義格式不一致,所以,不可以把函數(shù)名表示的地址設置給函數(shù)指針變量。我們來總結一下:
(1) 在Ubuntu系統(tǒng)中,使用GCC編譯,提示warning警告,但是,程序可以編譯通過,可以運行。
(2) 在Windows系統(tǒng)中,使用Visual Studio工具,無法編譯該代碼,提示類型不一致。
(3) 從代碼的嚴謹方面來說,是不可以設置類型不一致的數(shù)據。所以,我們應該編寫嚴謹?shù)拇a,函數(shù)定義的類型,與函數(shù)指針類型不一致的時候,不可以把函數(shù)名,賦值給函數(shù)指針變量。
函數(shù)指針變量的定義很重要,我們需要牢記和理解它們使用的方式。下面多舉幾個例子說明函數(shù)指針變量的定義和使用。
int func( void );
int (*pfunc)( void );
pfunc = func;
此時,定義func函數(shù),它的返回值類型是int類型,形參列表是void,那么,定義pfunc函數(shù)指針變量的時候,它的返回值類型與形參列表都必須與func一樣。
char * func1( int x, int y, int x);
char * (*pfunc1)( int , int , int );
pfunc1 = func1;
char * (*pfunc1)( int x, int y, int x);
我們再總結一下:
(1) 函數(shù)名表示函數(shù)在內存塊中的首地址,可以直接把函數(shù)名賦值給函數(shù)指針變量;
(2) 定義函數(shù)指針變量的時候,函數(shù)返回數(shù)據類型和形參列表必須與要指向函數(shù)的定義一致;
這個函數(shù)接受2個整形,m和n,
函數(shù)內容是一個循環(huán),進行n次,每次將p(初始值1)賦值為p*m,最后返回p的值。
比如x=p(2,3).
就是m=2,n=3.
運行3次循環(huán),第一次,p=p*m=1*2=2,
第二次,p=p*m=2*2=4,
第三次,p=p*m=4*2=8.
最后返回8,x=8.
C語言指向函數(shù)的指針承載的信息比較復雜,組織起來要素要寫全。根據指向函數(shù)的指針的書寫語法,下面的代碼就是一個返回函數(shù)指針的函數(shù):
int (*f(void))(int){//f是函數(shù),沒有參數(shù),返回指向一個函數(shù)的指針;這個被指向的函數(shù)有一個int型形參,且返回一個int值
return fun;//fun是個函數(shù)名,必須具備一個int形參、返回int值的要求
}
以下代碼可以驗證上述函數(shù)的正確性:
//#include?"stdafx.h"http://If?the?vc++6.0,?with?this?line.
#include?"stdio.h"
int?fun(int?n){
printf("abcdef\n");
return?n;
}
int?(*f(void))(int){
return?fun;
}
int?main(void){
printf("%d\n",f()(100));//調用函數(shù)f,通過f返回的指針用100調用函數(shù)fun
return?0;
}
輸出是:
abcdef
100
char *match (char*s,char ch1,char ch2)
函數(shù)各部分的含義:
第一個char,一般是代表函數(shù)的返回值是一個字符。但是它后面有一個*,應該把char和*連起來看,就是char*了。
那么,第一個就是char*,代表函數(shù)的返回值是一個字符指針(地址)。
match是函數(shù)的名字。
括號內是函數(shù)的參數(shù),多個參數(shù)用逗號分隔。本例共有3個參數(shù):char*s,char ch1,char ch2,第一個是s,類型為char*型(字符指針型),后兩個ch1、ch2都是char型(字符型)。
到這兒,你就知道去掉*match前的*為什么就不對了。
另外,括號里參數(shù)類型(如包含指針)跟函數(shù)返回值的類型無關。函數(shù)名前加不加*只與函數(shù)返回值的類型有關。