函數(shù)指針與指針函數(shù)的學(xué)習(xí)總結(jié)
函數(shù)指針是指向函數(shù)的指針,指針函數(shù)是指一個函數(shù)的返回值是一個指針,但下面的幾道題還是感覺很迷惑。各位能否講的詳細點呢?
(1) float(**def)[10] def是什么?
(2) double*(*gh)[10] gh是什么?
(3) double(*f[10])() f是什么?
(4) int*((*b)[10]) b是什么?這樣老感覺有點亂,有什么竅門可以記得并理解的清楚一點么?
======================
解答:
(1) def是一個指針, 指向的對象也是一個指針, 指向的指針最終指向的是10個float構(gòu)成的數(shù)組.
(2) gh是指針, 指向的是10個元素構(gòu)成的數(shù)組, 數(shù)組的元素是double*類型的指針.
(3) f是10個元素構(gòu)成的數(shù)組, 每個元素是指針, 指針指向的是函數(shù), 函數(shù)類型為無參數(shù)且返回值為double. 下面要講的竅門的例子跟這個很類似.
(4) b是指針,指向的是10個元素構(gòu)成的數(shù)組, 數(shù)組元素為int*類型的指針.
竅門如下:
如果我們碰到復(fù)雜的類型聲明,該如何解析它?例如:
char (*a[3])(int);
a到底被聲明為什么東東?指針?數(shù)組?還是函數(shù)?
分析時,從a 最接近(按運算符優(yōu)先級)處開始。我們看到a最接近符號是[ ]——注意:*比[ ]的優(yōu)先級低。a后既然有[ ],那么a是數(shù)組,而且是包含3個元素的數(shù)組。
那這個數(shù)組的每個元素是什么類型呢?雖然數(shù)組a只含有a[0]、a[1]、a[2]三個元素,a[3]實際上已經(jīng)越界,但在分析數(shù)組a的元素的類型時,我們正好需要形式上的元素a[3]。知道了a[3]的類型,就知道了a的元素的類型。 a[3]是什么類型?是指針,因為它的前面有*. 由此可知,數(shù)組a的元素是指針。
光說是指針還不夠。對于指針,必須說出它指向的東東是什么類型。它指向的東東是什么,就看*a[3]是什么(a[3]是指針,它指向的東東當(dāng)然是*a[3])了。繼續(xù)按優(yōu)先級觀察,我們看到*a[3]后面有小括號,所以可以肯定*a[3]是函數(shù)。即數(shù)組a的元素是指向函數(shù)的指針。
指向的是什么類型的函數(shù)?這很明顯,是入?yún)閕nt、返回值為char的類型的函數(shù)。
至此解析完畢。
按上述方法,再復(fù)雜的也可以一步步解析出來。
就像習(xí)武不是為了打人而是為了防身一樣,我們了解上述方法是為了看懂別人寫的復(fù)雜聲明,而不是為了在實踐中自己去構(gòu)造這種復(fù)雜的東東。實在需要復(fù)雜聲明時,可以用typedef替代一部分。例如上面語句可改成兩句:
typedef char (*FUN_PTR)(int);
FUN_PTR a[3];
這樣就清晰多了。
此外,上面的分析方法還讓我們對某些東西的本質(zhì)更加清楚。比如,n維數(shù)組的本質(zhì)都是一維數(shù)組。看個具體的例子:
int a[3][5];
這句聲明的是一個包含3個元素的一維數(shù)組,其每個元素又是一個由5個int數(shù)構(gòu)成的數(shù)組。我們不能理解為:a是一個包含5個元素的一維數(shù)組,其每個元素又是一個由3個int數(shù)構(gòu)成的數(shù)組。為什么?還是按上面的方法分析,這里從略。
有的書上或網(wǎng)上提供"向右看,向左看"的方法, 其實缺乏通用性, 比如它不適用于對多維數(shù)組本質(zhì)的分析. 而且這種方法掩蓋了本質(zhì). 本質(zhì)應(yīng)該是按上面所講的,根據(jù)運算符優(yōu)先級逐層剝開.
==============================================================================
一、指針函數(shù)
當(dāng)一個函數(shù)聲明其返回值為一個指針時,實際上就是返回一個地址給調(diào)用函數(shù),以用于需要指針或地址的表達式中。
格式:
類型說明符 * 函數(shù)名(參數(shù))
當(dāng)然了,由于返回的是一個地址,所以類型說明符一般都是int。
例如:int *GetDate();
int * aaa(int,int);
函數(shù)返回的是一個地址值,經(jīng)常使用在返回數(shù)組的某一元素地址上。
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];
}
程序應(yīng)該是很好理解的,子函數(shù)返回的是數(shù)組某元素的地址。輸出的是這個地址里的值。
二、函數(shù)指針
指向函數(shù)的指針包含了函數(shù)的地址,可以通過它來調(diào)用函數(shù)。聲明格式如下:
類型說明符 (*函數(shù)名)(參數(shù))
其實這里不能稱為函數(shù)名,應(yīng)該叫做指針的變量名。這個特殊的指針指向一個返回整型值的函數(shù)。指針的聲明必須和它指向函數(shù)的聲明保持一致。
指針名和指針運算符外面的括號改變了默認的運算符優(yōu)先級。如果沒有圓括號,就變成了一個返回整型指針的函數(shù)的原型聲明。
例如:
void (*fptr)();
把函數(shù)的地址賦值給函數(shù)指針,可以采用下面兩種形式:
fptr=&Function;
fptr=Function;
取地址運算符&不是必需的,因為單單一個函數(shù)標(biāo)識符就標(biāo)號表示了它的地址,如果是函數(shù)調(diào)用,還必須包含一個圓括號括起來的參數(shù)表。
可以采用如下兩種方式來通過指針調(diào)用函數(shù):
x=(*fptr)();
x=fptr();
第二種格式看上去和函數(shù)調(diào)用無異。但是有些程序員傾向于使用第一種格式,因為它明確指出是通過指針而非函數(shù)名來調(diào)用函數(shù)的。下面舉一個例子:
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;
如果有三個星號,那就是指針的指針的指針,四個星號就是指針的指針的指針的指針,依次類推。
當(dāng)你熟悉了簡單的例子以后,就可以應(yīng)付復(fù)雜的情況了。當(dāng)然,實際程序中,一般也只用到二級指針,三個星號不常見,更別說四個星號了。
指針的指針需要用到指針的地址。
char c='A';
char *p=&c;
char **cp=&p;
通過指針的指針,不僅可以訪問它指向的指針,還可以訪問它指向的指針?biāo)赶虻臄?shù)據(jù)。下面就是幾個這樣的例子:
char *p1=*cp; // (&c)
char c1=**cp;
你可能想知道這樣的結(jié)構(gòu)有什么用?利用指針的指針可以允許被調(diào)用函數(shù)修改局部指針變量和處理指針數(shù)組。
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)++;
}
首先用一個數(shù)組的地址初始化指針fp,然后把該指針的地址作為實參傳遞給函數(shù)FindCredit()。FindCredit()函數(shù)通過表達式**fpp間接地得到數(shù)組中的數(shù)據(jù)。
為遍歷數(shù)組以找到一個負值,F(xiàn)indCredit()函數(shù)進行自增運算的對象是調(diào)用者的指向數(shù)組的指針,而不是它自己的指向調(diào)用者指針的指針。語句(*fpp)++就是對形參指針指向的指針進行自增運算的。但是因為*運算符高于++運算符,所以圓括號在這里是必須的,如果沒有圓括號,那么++運算符將作用于二重指針fpp上。
四、指向指針數(shù)組的指針
指針的指針另一用法舊處理指針數(shù)組。有些程序員喜歡用指針數(shù)組來代替多維數(shù)組,一個常見的用法就是處理字符串。
char *Names[]=
{
Bill,
Sam,
Jim,
Paul,
Charles,
0
};
main()
{
char **nm=Names;
while(*nm!=0) printf(%s/n,*nm++);
}
先用字符型指針數(shù)組Names的地址來初始化指針nm。每次printf()的調(diào)用都首先傳遞指針nm指向的字符型指針,然后對nm進行自增運算使其指向數(shù)組的下一個元素(還是指針)。注意完成上述認為的語法為*nm++,它首先取得指針指向的內(nèi)容,然后使指針自增。
注意數(shù)組中的最后一個元素被初始化為0,while循環(huán)以次來判斷是否到了數(shù)組末尾。具有零值的指針常常被用做循環(huán)數(shù)組的終止符。程序員稱零值指針為空指針(NULL)。采用空指針作為終止符,在樹種增刪元素時,就不必改動遍歷數(shù)組的代碼,因為此時數(shù)組仍然以空指針作為結(jié)束。
上一篇:VC6.0常見鏈接錯誤與解決方法
欄 目:C語言
下一篇:C語言函數(shù)的遞歸和調(diào)用實例分析
本文標(biāo)題:函數(shù)指針與指針函數(shù)的學(xué)習(xí)總結(jié)
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/4290.html
您可能感興趣的文章
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用函數(shù)刪除字符
- 04-02c語言的正則匹配函數(shù) c語言正則表達式函數(shù)庫
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對數(shù)怎么表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段函數(shù)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排序法函數(shù)
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段函數(shù)
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求階乘


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機閱讀
- 01-10delphi制作wav文件的方法
- 01-11ajax實現(xiàn)頁面的局部加載
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 04-02jquery與jsp,用jquery
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-10C#中split用法實例總結(jié)
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改


