1、
在學(xué)習(xí)arm進程中發(fā)現(xiàn)這“指針函數(shù)”與“函數(shù)指針”容易弄錯,所以今天,我自己想1次把它弄清楚,找了1些資料,首先它們之間的定義:
1、指針函數(shù)是指帶指針的函數(shù),即本質(zhì)是1個函數(shù)。函數(shù)返回類型是某1類型的指針
類型標(biāo)識符 *函數(shù)名(參數(shù)表)
int *f(x,y);
首先它是1個函數(shù),只不過這個函數(shù)的返回值是1個地址值。函數(shù)返回值必須用同類型的指針變量來接受,也就是說,指針函數(shù)1定有函數(shù)返回值,而且,在主調(diào)函數(shù)中,函數(shù)返回值必須賦給同類型的指針變量。
表示:
float *fun();
float *p;
p = fun(a);
注意指針函數(shù)與函數(shù)指針表示方法的不同,千萬不要混淆。最簡單的辨別方式就是看函數(shù)名前面的指針*號有無被括號()包括,如果被包括就是函數(shù)指針,反之則是指針函數(shù)。
來說詳細(xì)1些吧!請看下面
指針函數(shù):
當(dāng)1個函數(shù)聲明其返回值為1個指針時,實際上就是返回1個地址給調(diào)用函數(shù),以用于需要指針或地址的表達式中。
格式:
類型說明符 * 函數(shù)名(參數(shù))
固然了,由于返回的是1個地址,所以類型說明符1般都是int。
例如:int *GetDate();
int * aaa(int,int);
函數(shù)返回的是1個地址值,常常使用在返回數(shù)組的某1元素地址上。
int * GetDate(int wk,int dy);
main()
{
int wk,dy;
do
{
printf(Enter week(1⑸)day(1⑺)
);
scanf(%d%d,&wk,&dy);
}
while(wk<1||wk>5||dy<1||dy>7);
printf(%d
,*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,⑴}
};
return &calendar[wk⑴][dy⑴];
}
程序應(yīng)當(dāng)是很好理解的,子函數(shù)返回的是數(shù)組某元素的地址。輸出的是這個地址里的值。
2、函數(shù)指針是指向函數(shù)的指針變量,即本質(zhì)是1個指針變量。
int (*f) (int x); /* 聲明1個函數(shù)指針 */
f=func; /* 將func函數(shù)的首地址賦給指針f */
指向函數(shù)的指針包括了函數(shù)的地址,可以通過它來調(diào)用函數(shù)。聲明格式以下:
類型說明符 (*函數(shù)名)(參數(shù))
其實這里不能稱為函數(shù)名,應(yīng)當(dāng)叫做指針的變量名。這個特殊的指針指向1個返回整型值的函數(shù)。指針的聲明筆削和它指向函數(shù)的聲明保持1致。
指針名和指針運算符外面的括號改變了默許的運算符優(yōu)先級。如果沒有圓括號,就變成了1個返回整型指針的函數(shù)的原型聲明。
例如:
void (*fptr)();
把函數(shù)的地址賦值給函數(shù)指針,可以采取下面兩種情勢:
fptr=&Function;
fptr=Function;
取地址運算符&不是必須的,由于單單1個函數(shù)標(biāo)識符就標(biāo)號表示了它的地址,如果是函數(shù)調(diào)用,還必須包括1個圓括號括起來的參數(shù)表。
可以采取以下兩種方式來通過指針調(diào)用函數(shù):
x=(*fptr)();
x=fptr();
第2種格式看上去和函數(shù)調(diào)用無異。但是有些http://www.vxbq.cn偏向于使用第1種格式,由于它明確指出是通過指針而非函數(shù)名來調(diào)用函數(shù)的。下面舉1個例子:
void (*funcp)();
void FileFunc(),EditFunc();
main()
{
funcp=FileFunc;
(*funcp)();
funcp=EditFunc;
(*funcp)();
}
void FileFunc()
{
printf(FileFunc
);
}
void EditFunc()
{
printf(EditFunc
);
}
程序輸出為:
FileFunc
EditFunc
主要的區(qū)分是1個是指針變量,1個是函數(shù)。在使用是必要要弄清楚才能正確使用
2、指針的指針
指針的指針看上去有些使人費解。它們的聲明有兩個星號。例如:
char ** cp;
如果有3個星號,那就是指針的指針的指針,4個星號就是指針的指針的指針的指針,順次類推。當(dāng)你熟習(xí)了簡單的例子以后,就能夠應(yīng)付復(fù)雜的情況了。固然,實際程序中,1般也只用到 2級指針,3個星號不常見,更別說4個星號了。
指針的指針需要用到指針的地址。
char c='A';
char *p=&c;
char **cp=&p;
通過指針的指針,不但可以訪問它指向的指針,還可以訪問它指向的指針?biāo)赶虻臄?shù)據(jù)。下面就是幾個這樣的例子:
char *p1=*cp;
char c1=**cp;
你可能想知道這樣的結(jié)構(gòu)有甚么用。利用指針的指針可以允許被調(diào)用函數(shù)修改局部指針變量和處理指針數(shù)組。
void FindCredit(int **);
main()
{
int vals[]={7,6,5,⑷,3,2,1,0};
int *fp=vals;
FindCredit(&fp);
printf(%d
,*fp);
}
void FindCredit(int ** fpp)
{
while(**fpp!=0)
if(**fpp<0) break;
else (*fpp)++;
}
首先用1個數(shù)組的地址初始化指針fp,然后把該指針的地址作為實參傳遞給函數(shù)FindCredit()。FindCredit()函數(shù)通過表達式**fpp間接地得到數(shù)組中的數(shù)據(jù)。為遍歷數(shù)組以找到1個負(fù)值,F(xiàn)indCredit()函數(shù)進行自增運算的對象是調(diào)用者的指向數(shù)組的指針,而不是它自己的指向調(diào)用者指針的指針。語句(*fpp)++就是對形參指針指向的指針進行自增運算的。但是由于*運算符高于++運算符,所以圓括號在這里是必須的,如果沒有圓括號,那末++運算符將作用于2重指針fpp上。
3、指向指針數(shù)組的指針
指針的指針另外一用法舊處理指針數(shù)組。有些http://www.vxbq.cn喜歡用指針數(shù)組來代替多維數(shù)組,1個常見的用法就是處理字符串。
char *Names[]=
{
Bill,
Sam,
Jim,
Paul,
Charles,
0
};
main()
{
char **nm=Names;
while(*nm!=0) printf(%s
,*nm++);
}
先用字符型指針數(shù)組Names的地址來初始化指針nm。每次printf()的調(diào)用都首先傳遞指針nm指向的字符型指針,然后對nm進行自增運算使其指向數(shù)組的下1個元素(還是指針)。注意完成上述認(rèn)為的語法為*nm++,它首先獲得指針指向的內(nèi)容,然后使指針自增。
注意數(shù)組中的最后1個元素被初始化為0,while循環(huán)以次來判斷是不是到了數(shù)組末尾。具有零值的指針常常被用做循環(huán)數(shù)組的終止符。http://www.vxbq.cn稱零值指針為空指針(NULL)。采取空指針作為終止符,在樹種增刪元素時,就沒必要改動遍歷數(shù)組的代碼,由于此時數(shù)組依然以空指針作為結(jié)束。