深入理解指針函數(shù)
顧名思義,指針函數(shù)即返回指針的函數(shù)。其一般定義形式如下:
類型名 *函數(shù)名(函數(shù)參數(shù)表列);
其中,后綴運算符括號“()”表示這是一個函數(shù),其前綴運算符星號“*”表示此函數(shù)為指針型函數(shù),其函數(shù)值為指針,即它帶回來的值的類型為指針,當(dāng)調(diào)用這個函數(shù)后,將得到一個“指向返回值為…的指針(地址),“類型名”表示函數(shù)返回的指針指向的類型”。
“(函數(shù)參數(shù)表列)”中的括號為函數(shù)調(diào)用運算符,在調(diào)用語句中,即使函數(shù)不帶參數(shù),其參數(shù)表的一對括號也不能省略。其示例如下:
int *pfun(int, int);
由于“*”的優(yōu)先級低于“()”的優(yōu)先級,因而pfun首先和后面的“()”結(jié)合,也就意味著,pfun是一個函數(shù)。即:
int *(pfun(int, int));
接著再和前面的“*”結(jié)合,說明這個函數(shù)的返回值是一個指針。由于前面還有一個int,也就是說,pfun是一個返回值為整型指針的函數(shù)。
我們不妨來再看一看,指針函數(shù)與函數(shù)指針有什么區(qū)別?
int (*pfun)(int, int);
通過括號強行將pfun首先與“*”結(jié)合,也就意味著,pfun是一個指針,接著與后面的“()”結(jié)合,說明該指針指向的是一個函數(shù),然后再與前面的int結(jié)合,也就是說,該函數(shù)的返回值是int。由此可見,pfun是一個指向返回值為int的函數(shù)的指針。
雖然它們只有一個括號的差別,但是表示的意義卻截然不同。函數(shù)指針的本身是一個指針,指針指向的是一個函數(shù)。指針函數(shù)的本身是一個函數(shù),其函數(shù)的返回值是一個指針。
2. 用函數(shù)指針作為函數(shù)的返回值
在上面提到的指針函數(shù)里面,有這樣一類函數(shù),它們也返回指針型數(shù)據(jù)(地址),但是這個指針不是指向int、char之類的基本類型,而是指向函數(shù)。對于初學(xué)者,別說寫出這樣的函數(shù)聲明,就是看到這樣的寫法也是一頭霧水。比如,下面的語句:
int (*ff(int))(int *, int);
我們用上面介紹的方法分析一下,ff首先與后面的“()”結(jié)合,即:
int (*(ff(int)))(int *, int); // 用括號將ff(int)再括起來
也就意味著,ff是一個函數(shù)。
接著與前面的“*”結(jié)合,說明ff函數(shù)的返回值是一個指針。然后再與后面的“()”結(jié)合,也就是說,該指針指向的是一個函數(shù)。
這種寫法確實讓人非常難懂,以至于一些初學(xué)者產(chǎn)生誤解,認(rèn)為寫出別人看不懂的代碼才能顯示自己水平高。而事實上恰好相反,能否寫出通俗易懂的代碼是衡量程序員是否優(yōu)秀的標(biāo)準(zhǔn)。一般來說,用typedef關(guān)鍵字會使該聲明更簡單易懂。在前面我們已經(jīng)見過:
int (*PF)(int *, int);
也就是說,PF是一個函數(shù)指針“變量”。當(dāng)使用typedef聲明后,則PF就成為了一個函數(shù)指針“類型”,即:
typedef int (*PF)(int *, int);
這樣就定義了返回值的類型。然后,再用PF作為返回值來聲明函數(shù):
PF ff(int);
下面將以程序清單1為例,說明用函數(shù)指針作為函數(shù)的返回值的用法。當(dāng)程序接收用戶輸入時,如果用戶輸入d,則求數(shù)組的最大值,如果輸入x,則求數(shù)組的最小值,如果輸入p,則求數(shù)組的平均值。
程序清單 1 求最值與平均值示例
1 #include
2 #include
3 double GetMin(double *dbData, int iSize) // 求最小值
4 {
5 double dbMin;
6 int i;
7
8 assert(iSize>0);
9 dbMin=dbData[0];
10 for (i=1; i
11 if (dbMin>dbData[i]) {
12 dbMin=dbData[i];
13 }
14 }
15 return dbMin;
16 }
17
18 double GetMax(double *dbData, int iSize) // 求最大值
19 {
20 double dbMax;
21 int i;
22
23 assert(iSize>0);
24 dbMax=dbData[0];
25 for (i=1; i
26 if (dbMax< dbData[i]) {
27 dbMax=dbData[i];
28 }
29 }
30 return dbMax;
31 }
32
33 double GetAverage(double *dbData, int iSize) // 求平均值
34 {
35 double dbSum=0;
36 int i;
37
38 assert(iSize>0);
39 for (i=0; i
40 {
41 dbSum+=dbData[i];
42 }
43 return dbSum/iSize;
44 }
45
46 double UnKnown(double *dbData, int iSize) // 未知算法
47 {
48 return 0;
49 }
50
51 typedef double (*PF)(double *dbData, int iSize); // 定義函數(shù)指針類型
52 PF GetOperation(char c) // 根據(jù)字符得到操作類型,返回函數(shù)指針
53 {
54 switch (c)
55 {
56 case d:
57 return GetMax;
58 case x:
59 return GetMin;
60 case p:
61 return GetAverage;
62 default:
63 return UnKnown;
64 }
65 }
66
67 int main(void)
68 {
69 double dbData[]={3.1415926, 1.4142, -0.5,999, -313, 365};
70 int iSize=sizeof(dbData)/sizeof(dbData[0]);
71 char c;
72
73 printf("Please input the Operation :n");
74 c=getchar();
75 printf("result is %lfn", GetOperation(c)(dbData,iSize)); // 通過函數(shù)指針調(diào)用函數(shù)
76 }
上述程序中前面4個函數(shù)分別實現(xiàn)求最大值、最小值、平均值和未知算法,然后實現(xiàn)了GetOperation函數(shù)。這個函數(shù)根據(jù)字符的返回值實現(xiàn)上面4個函數(shù)。它是以函數(shù)指針的形式返回的,從后面的main函數(shù)的GetOperation(c)(dbData, iSize)可以看出,通過這個指針可以調(diào)用函數(shù)。
關(guān)鍵詞:
指針函
評論