整個(gè)程序的目的是對(duì)一組數(shù)據(jù)進(jìn)行求最大值、最小值或平均值的操作,通過(guò)輸入的指令進(jìn)行相應(yīng)的操作。按照不用函數(shù)指針的思路,程序大概會(huì)寫成這樣:(以下為偽代碼)
成都創(chuàng)新互聯(lián)公司是工信部頒發(fā)資質(zhì)IDC服務(wù)器商,為用戶提供優(yōu)質(zhì)的綿陽(yáng)主機(jī)托管服務(wù)
最大值函數(shù){...}
最小值函數(shù){...}
平均值函數(shù){...}
main{
輸入d,則調(diào)用 最大值函數(shù)
輸入x,則調(diào)用 最小值函數(shù)
輸入p,則調(diào)用平均值函數(shù)
}
不過(guò)可以發(fā)現(xiàn)如下幾點(diǎn):
· 三個(gè)函數(shù)的參數(shù)形式與返回值都是完全一樣的
· 按上述方法寫代碼,每次調(diào)用都要寫清楚參數(shù),略顯啰嗦
所以就使用函數(shù)指針來(lái)重新改寫代碼
GetOperation這個(gè)函數(shù)本身,就是根據(jù)輸入指令找到對(duì)應(yīng)的 操作函數(shù)并作為函數(shù)指針返回它,然后main里直接通過(guò)函數(shù)指針調(diào)用,調(diào)用代碼只需要寫一次,而且甚至不用關(guān)心這個(gè)函數(shù)指針代表的到底是哪個(gè)函數(shù)
這個(gè)程序里只有三個(gè)對(duì)數(shù)據(jù)的操作所以函數(shù)指針的優(yōu)點(diǎn)不是很明顯。假設(shè)有多大100種對(duì)數(shù)據(jù)的操作,如果不使用函數(shù)指針,就要不停地寫if else if else if...,使用函數(shù)指針,甚至可以事先建立一個(gè)指令字符與操作函數(shù)指針的對(duì)應(yīng)列表,運(yùn)行時(shí)直接在列表里搜索抽取,代碼就顯得干凈利索
GetOperation(c)(dbData,iSize)這句話的具體執(zhí)行操作是:
(下面假設(shè)輸入了c='d')
首先執(zhí)行GetOperation('d'),該函數(shù)返回得到GetMax(的指針),再執(zhí)行GetMax(bData,iSize),即對(duì)數(shù)據(jù)求最大值的操作,函數(shù)返回得到最大值
#includestdio.h
int *d_bubblesort(int a[],int n)//數(shù)組長(zhǎng)度為n的冒泡降序排序
{
int i,j;
int temp;
for(i=0;in;i++)
{
for(j=n-1;ji;j--)
{
if(a[j]a[j-1])
{
temp=a[j-1];
a[j-1]=a[j];
a[j]=temp;
}
}
}
return a;
}
void main()
{
int i;
int *p=NULL;
int a[10]={6,5,4,1,2,3,9,8,7,10};
int *(*fun)(int*,int);
fun=d_bubblesort;
p=fun(a,10);
for(i=0;i10;i++)
{
printf("%d\t",p[i]);
}
}
并不是不規(guī)范的問(wèn)題,你子程序?qū)懛ㄖ荒茉谧映绦蛑休敵稣_的數(shù)據(jù),而在主函數(shù)中無(wú)法得到交換過(guò)的x和y的值。參考程序運(yùn)用指針讓子程序去讀寫a和b的值,在子程序執(zhí)行完成后,a和b的值確實(shí)得到了交換。
你的程序中子程序采用的不是指針類型參數(shù),這樣子程序無(wú)法將交換過(guò)后的值傳回來(lái)。
函數(shù)在內(nèi)存中有一個(gè)物理位置,而這個(gè)位置是可以賦給一個(gè)指針的。一零點(diǎn)函數(shù)的地址就是該函數(shù)的入口點(diǎn)。因此,函數(shù)指針可被用來(lái)調(diào)用一個(gè)函數(shù)。函數(shù)的地址是用不帶任何括號(hào)或參數(shù)的函數(shù)名來(lái)得到的。(這很類似于數(shù)組地址的得到方法,即,在只有數(shù)組名而無(wú)下標(biāo)是就得到數(shù)組地址。)
怎樣說(shuō)明一個(gè)函數(shù)指針變量呢 ?
為了說(shuō)明一個(gè)變量 fn_pointer 的類型是"返回值為 int 的函數(shù)指針", 你可以使用下面的說(shuō)明語(yǔ)句:
int (*fn_pointer) ();
為了讓編譯器能正確地解釋這句語(yǔ)句, *fn_pointer 必須用括號(hào)圍起來(lái)。若漏了這對(duì)括號(hào), 則:
int *fn_pointer ();
的意思完全不同了。fn_pointer 將是一個(gè)函數(shù)名, 其返回值為 int 類型的指針。
2:函數(shù)指針變量
在C語(yǔ)言中規(guī)定,一個(gè)函數(shù)總是占用一段連續(xù)的內(nèi)存區(qū), 而函數(shù)名就是該函數(shù)所占內(nèi)存區(qū)的首地址。 我們可以把函數(shù)的這個(gè)首地址 ( 或稱入口地址 ) 賦予一個(gè)指針變量, 使該指針變量指向該函數(shù)。然后通過(guò)指針變量就可以找到并調(diào)用這個(gè)函數(shù)。我們把這種指向函數(shù)的指針變量稱為 " 函數(shù)指針變量 " 。
函數(shù)指針變量定義的一般形式為:
類型說(shuō)明符 (* 指針變量名 )();
其中 " 類型說(shuō)明符 " 表示被指函數(shù)的返回值的類型。 "(* 指針變量名 )" 表示 "*" 后面的變量是定義的指針變量。 最后的空括號(hào)表示指針變量所指的是一個(gè)函數(shù)。
例如: int (*pf)();
表示 pf 是一個(gè)指向函數(shù)入口的指針變量,該函數(shù)的返回值 ( 函數(shù)值 ) 是整型。
下面通過(guò)例子來(lái)說(shuō)明用指針形式實(shí)現(xiàn)對(duì)函數(shù)調(diào)用的方法。
int max(int a,int b)
{
if(ab)return a;
else return b;
}
main()
{
int max(int a,int b);
int(*pmax)();
int x,y,z;
pmax=max;
printf("input two numbers:/n");
scanf("%d%d",x,y);
z=(*pmax)(x,y);
printf("maxmum=%d",z);
}
從上述程序可以看出用,函數(shù)指針變量形式調(diào)用函數(shù)的步驟如下:
1. 先定義函數(shù)指針變量,如后一程序中第 9 行 int (*pmax)(); 定義 pmax 為函數(shù)指針變量。
2. 把被調(diào)函數(shù)的入口地址 ( 函數(shù)名 ) 賦予該函數(shù)指針變量,如程序中第 11 行 pmax=max;
3. 用函數(shù)指針變量形式調(diào)用函數(shù),如程序第 14 行 z=(*pmax)(x,y); 調(diào)用函數(shù)的一般形式為: (* 指針變量名 ) ( 實(shí)參表 ) 使用函數(shù)指針變量還應(yīng)注意以下兩點(diǎn):
a. 函數(shù)指針變量不能進(jìn)行算術(shù)運(yùn)算,這是與數(shù)組指針變量不同的。數(shù)組指針變量加減一個(gè)整數(shù)可使指針移動(dòng)指向后面或前面的數(shù)組元素,而函數(shù)指針的移動(dòng)是毫無(wú)意義的。
b. 函數(shù)調(diào)用中 "(* 指針變量名 )" 的兩邊的括號(hào)不可少,其中的 * 不應(yīng)該理解為求值運(yùn)算,在此處它只是一種表示符號(hào)。
3:指針型函數(shù)
前面我們介紹過(guò),所謂函數(shù)類型是指函數(shù)返回值的類型。 在C語(yǔ)言中允許一個(gè)函數(shù)的返回值是一個(gè)指針 ( 即地址 ) ,這種返回指針值的函數(shù)稱為指針型函數(shù)。
定義指針型函數(shù)的一般形式為:
類型說(shuō)明符 * 函數(shù)名 ( 形參表 )
{
…… /* 函數(shù)體 */
}
其中函數(shù)名之前加了 "*" 號(hào)表明這是一個(gè)指針型函數(shù),即返回值是一個(gè)指針。類型說(shuō)明符表示了返回的指針值所指向的數(shù)據(jù)類型。
如:
int *ap(int x,int y)
{
…… /* 函數(shù)體 */
}
表示 ap 是一個(gè)返回指針值的指針型函數(shù), 它返回的指針指向一個(gè)整型變量。下例中定義了一個(gè)指針型函數(shù) day_name ,它的返回值指向一個(gè)字符串。該函數(shù)中定義了一個(gè)靜態(tài)指針數(shù)組 name 。 name 數(shù)組初始化賦值為八個(gè)字符串,分別表示各個(gè)星期名及出錯(cuò)提示。形參 n 表示與星期名所對(duì)應(yīng)的整數(shù)。在主函數(shù)中, 把輸入的整數(shù) i 作為實(shí)參, 在 printf 語(yǔ)句中調(diào)用 day_name 函數(shù)并把 i 值傳送給形參 n 。 day_name 函數(shù)中的 return 語(yǔ)句包含一個(gè)條件表達(dá)式, n 值若大于 7 或小于 1 則把 name[0] 指針返回主函數(shù)輸出出錯(cuò)提示字符串 "Illegal day" 。否則返回主函數(shù)輸出對(duì)應(yīng)的星期名。主函數(shù)中的第 7 行是個(gè)條件語(yǔ)句,其語(yǔ)義是,如輸入為負(fù)數(shù) (i0) 則中止程序運(yùn)行退出程序。 exit 是一個(gè)庫(kù)函數(shù), exit(1) 表示發(fā)生錯(cuò)誤后退出程序, exit(0) 表示正常退出。
應(yīng)該特別注意的是函數(shù)指針變量和指針型函數(shù)這兩者在寫法和意義上的區(qū)別。如 int(*p)() 和 int *p() 是兩個(gè)完全不同的量。 int(*p)() 是一個(gè)變量說(shuō)明,說(shuō)明 p 是一個(gè)指向函數(shù)入口的指針變量,該函數(shù)的返回值是整型量, (*p) 的兩邊的括號(hào)不能少。
int *p() 則不是變量說(shuō)明而是函數(shù)說(shuō)明,說(shuō)明 p 是一個(gè)指針型函數(shù),其返回值是一個(gè)指向整型量的指針,*p 兩邊沒有括號(hào)。作為函數(shù)說(shuō)明, 在括號(hào)內(nèi)最好寫入形式參數(shù),這樣便于與變量說(shuō)明區(qū)別。 對(duì)于指針型函數(shù)定義,int *p() 只是函數(shù)頭部分,一般還應(yīng)該有函數(shù)體部分。
main()
{
int i;
char *day_name(int n);
printf("input Day No:/n");
scanf("%d",i);
if(i0) exit(1);
printf("Day No:%2d--%s/n",i,day_name(i));
}
char *day_n
ame(int n)
{
static char *name[]={ "Illegal day",
"Monday",
"Tuesday",
"Wednesday",
"Thursday",
"Friday",
"Saturday",
"Sunday"};
return((n1||n7) ? name[0] : name[n]);
}
本程序是通過(guò)指針函數(shù),輸入一個(gè) 1 ~ 7 之間的整數(shù), 輸出對(duì)應(yīng)的星期名。指針數(shù)組的說(shuō)明與使用一個(gè)數(shù)組的元素值為指針則是指針數(shù)組。指針數(shù)組是一組有序的指針的集合。指針數(shù)組的所有元素都必須是具有相同存儲(chǔ)類型和指向相同數(shù)據(jù)類型的指針變量。
指針數(shù)組說(shuō)明的一般形式為: 類型說(shuō)明符 * 數(shù)組名 [ 數(shù)組長(zhǎng)度 ]
其中類型說(shuō)明符為指針值所指向的變量的類型。例如: int *pa[3] 表示 pa 是一個(gè)指針數(shù)組,它有三個(gè)數(shù)組元素, 每個(gè)元素值都是一個(gè)指針,指向整型變量。通常可用一個(gè)指針數(shù)組來(lái)指向一個(gè)二維數(shù)組。 指針數(shù)組中的每個(gè)元素被賦予二維數(shù)組每一行的首地址,因此也可理解為指向一個(gè)一維數(shù)組。圖 6—6 表示了這種關(guān)系。
int a[3][3]={1,2,3,4,5,6,7,8,9};
int *pa[3]={a[0],a[1],a[2]};
int *p=a[0];
main()
{
int i;
for(i=0;i3;i++)
printf("%d,%d,%d/n",a[i][2-i],*a[i],*(*(a+i)+i));
for(i=0;i3;i++)
printf("%d,%d,%d/n",*pa[i],p[i],*(p+i));
}
本例程序中, pa 是一個(gè)指針數(shù)組,三個(gè)元素分別指向二維數(shù)組 a 的各行。然后用循環(huán)語(yǔ)句輸出指定的數(shù)組元素。其中 *a[i] 表示 i 行 0 列元素值; *(*(a+i)+i) 表示 i 行 i 列的元素值; *pa[i] 表示 i 行 0 列元素值;由于 p 與 a[0] 相同,故 p[i] 表示 0 行 i 列的值; *(p+i) 表示 0 行 i 列的值。讀者可仔細(xì)領(lǐng)會(huì)元素值的各種不同的表示方法。 應(yīng)該注意指針數(shù)組和二維數(shù)組指針變量的區(qū)別。 這兩者雖然都可用來(lái)表示二維數(shù)組,但是其表示方法和意義是不同的