函數指標是指向函數的指標,指標函數是指一個函數的傳回值是一個指標,但下面的幾道題還是感覺很迷惑。各位能否講的詳細點呢?
(1) float(**def)[10] def是什嗎?
(2) double*(*gh)[10] gh是什嗎?
(3) double(*f[10])() f是什嗎?
(4) int*((*b)[10]) b是什嗎?這樣老感覺有點亂,有什麼竅門可以記得並理解的清楚一點嗎?
======================
解答:
(1) def是一個指標, 指向的對象也是一個指標, 指向的指標最終指向的是10個float構成的數組.
(2) gh是指標, 指向的是10個元素構成的數組, 數組的元素是double*類型的指標.
(3) f是10個元素構成的數組, 每個元素是指標, 指標指向的是函數, 函數類型為無參數且傳回值為double. 下面要講的竅門的例子跟這個很類似.
(4) b是指標,指向的是10個元素構成的數組, 數組元素為int*類型的指標.
竅門如下:
如果我們碰到複雜的型別宣告,該如何解析它?例如:
char (*a[3])(int);
a到底被聲明為什麼東東?指標?數組?還是函數?
分析時,從a 最接近(按運算子優先順序)處開始。我們看到a最接近符號是[ ]——注意:*比[ ]的優先順序低。a後既然有[ ],那麼a是數組,而且是包含3個元素的數組。
那 這個數組的每個元素是什麼類型呢?雖然數組a只含有a[0]、a[1]、a[2]三個元素,a[3]實際上已經越界,但在分析數組a的元素的類型時,我們 正好需要形式上的元素a[3]。知道了a[3]的類型,就知道了a的元素的類型。 a[3]是什麼類型?是指標,因為它的前面有*. 由此可知,數組a的元素是指標。
光說是指標還不夠。對於指標,必須說出它指向的東東是什麼類型。它指向的東東是什麼,就看*a[3]是什 麼(a[3]是指標,它指向的東東當然是*a[3])了。繼續按優先順序觀察,我們看到*a[3]後面有小括弧,所以可以肯定*a[3]是函數。即數組a的 元素是指向函數的指標。
指向的是什麼類型的函數?這很明顯,是入參為int、傳回值為char的類型的函數。
至此解析完畢。
按上述方法,再複雜的也可以一步步解析出來。
就像習武不是為了打人而是為了防身一樣,我們瞭解上述方法是為了看懂別人寫的複雜聲明,而不是為了在實踐中自己去構造這種複雜的東東。實在需要複雜聲明時,可以用typedef替代一部分。例如上面語句可改成兩句:
typedef char (*FUN_PTR)(int);
FUN_PTR a[3];
這樣就清晰多了。
此外,上面的分析方法還讓我們對某些東西的本質更加清楚。比如,n維數組的本質都是一維數組。看個具體的例子:
int a[3][5];
這句聲明的是一個包含3個元素的一維數組,其每個元素又是一個由5個int數構成的數組。我們不能理解為:a是一個包含5個元素的一維數組,其每個元素又是一個由3個int數構成的數組。為什嗎?還是按上面的方法分析,這裡從略。
有的書上或網上提供"向右看,向左看"的方法, 其實缺乏通用性, 比如它不適用於對多維陣列本質的分析. 而且這種方法掩蓋了本質. 本質應該是按上面所講的,根據運算子優先順序逐層剝開.
==============================================================================
一、指標函數
當一個函式宣告其傳回值為一個指標時,實際上就是返回一個地址給調用函數,以用於需要指標或地址的運算式中。
格式:
類型說明符 * 函數名(參數)
當然了,由於返回的是一個地址,所以類型說明符一般都是int。
例如:int *GetDate();
int * aaa(int,int);
函數返回的是一個地址值,經常使用在返回數組的某一元素地址上。
複製代碼 代碼如下:
int * GetDate(int wk,int dy);
main()
{
int wk,dy;
do
{
printf(Enter week(1-5)day(1-7)/n);
scanf(%d%d,&wk,&dy);
}
while(wk<1||wk>5||dy<1||dy>7);
printf(%d/n,*GetDate(wk,dy));
}
int * GetDate(int wk,int dy)
{
static int calendar[5][7]=
{
{1,2,3,4,5,6,7},
{8,9,10,11,12,13,14},
{15,16,17,18,19,20,21},
{22,23,24,25,26,27,28},
{29,30,31,-1}
};
return &calendar[wk-1][dy-1];
}
程式應該是很好理解的,子函數返回的是數組某元素的地址。輸出的是這個地址裡的值。
二、函數指標
指向函數的指標包含了函數的地址,可以通過它來調用函數。聲明格式如下:
類型說明符 (*函數名)(參數)
其實這裡不能稱為函數名,應該叫做指標的變數名。這個特殊的指標指向一個返回整型值的函數。指標的聲明必須和它指向函數的聲明保持一致。
指標名和指標運算子外面的括弧改變了預設的運算子優先順序。如果沒有圓括弧,就變成了一個返回整型指標的函數的原型聲明。
例如:
void (*fptr)();
把函數的地址賦值給函數指標,可以採用下面兩種形式:
fptr=&Function;
fptr=Function;
取地址運算子&不是必需的,因為單單一個函數標識符就標號表示了它的地址,如果是函數調用,還必須包含一個圓括弧括起來的參數表。
可以採用如下兩種方式來通過指標調用函數:
x=(*fptr)();
x=fptr();
第二種格式看上去和函數調用無異。但是有些程式員傾向於使用第一種格式,因為它明確指出是通過指標而非函數名來調用函數的。下面舉一個例子:
複製代碼 代碼如下:
void (*funcp)();
void FileFunc(),EditFunc();
main()
{
funcp=FileFunc;
(*funcp)();
funcp=EditFunc;
(*funcp)();
}
void FileFunc()
{
printf("FileFunc/n");
}
void EditFunc()
{
printf("EditFunc/n");
}
程式輸出為:
FileFunc
EditFunc
三、指標的指標
指標的指標看上去有些令人費解。它們的聲明有兩個星號。例如:
char ** cp;
如果有三個星號,那就是指標的指標的指標,四個星號就是指標的指標的指標的指標,依次類推。
當你熟悉了簡單的例子以後,就可以應付複雜的情況了。當然,實際程式中,一般也只用到二級指標,三個星號不常見,更別說四個星號了。
指標的指標需要用到指標的地址。
char c='A';
char *p=&c;
char **cp=&p;
通過指標的指標,不僅可以訪問它指向的指標,還可以訪問它指向的指標所指向的資料。下面就是幾個這樣的例子:
char *p1=*cp; // (&c)
char c1=**cp;
你可能想知道這樣的結構有什麼用?利用指標的指標可以允許被調用函數修改局部指標變數和處理指標數組。
複製代碼 代碼如下:
void FindCredit(int **);
main()
{
int vals[]={7,6,5,-4,3,2,1,0};
int *fp=vals;
FindCredit(&fp);
printf(%d/n,*fp);
}
void FindCredit(int ** fpp)
{
while(**fpp!=0)
if(**fpp<0) break;
else (*fpp)++;
}
首先用一個數組的地址初始化指標fp,然後把該指標的地址作為實參傳遞給函數FindCredit()。FindCredit()函數通過運算式**fpp間接地得到數組中的資料。
為 遍曆數組以找到一個負值,FindCredit()函數進行自增運算的對象是調用者的指向數組的指標,而不是它自己的指向調用者指標的指標。語句 (*fpp)++就是對形參指標指向的指標進行自增運算的。但是因為*運算子高於++運算子,所以圓括弧在這裡是必須的,如果沒有圓括弧,那麼++運算子 將作用於二重指標fpp上。
四、指向指標數組的指標
指標的指標另一用法舊處理指標數組。有些程式員喜歡用指標數組來代替多維陣列,一個常見的用法就是處理字串。
複製代碼 代碼如下:
char *Names[]=
{
Bill,
Sam,
Jim,
Paul,
Charles,
0
};
main()
{
char **nm=Names;
while(*nm!=0) printf(%s/n,*nm++);
}
先用字元型指標數組Names的地址來初始化指標nm。每次printf()的調用都首先傳遞指標nm指向的字元型指標,然後對nm進行自增運算使其指向數組的下一個元素(還是指標)。注意完成上述認為的文法為*nm++,它首先取得指標指向的內容,然後使指標自增。
注 意數組中的最後一個元素被初始化為0,while迴圈以次來判斷是否到了數組末尾。具有零值的指標常常被用做迴圈數組的終止符。程式員稱零值指標為空白指標 (NULL)。採用null 指標作為終止符,在樹種增刪元素時,就不必改動遍曆數組的代碼,因為此時數組仍然以null 指標作為結束。