一個函數指針必須用* 操作符(和一對額外的括弧) “轉換為” 一個“真正的” 函數才能調用:
int r, func(), (*fp)() = func;
r = (*fp)();
而函數總是通過指針進行調用的, 所有“真正的” 函數名總是隱式的退化為指針(在表達式中, 正如在初始化時一樣。參見問題1.14)。這個推論表明無論fp 是函數名和函數的指針r = fp();
ansi c 標準實際上接受後邊的解釋, 這意味着* 操作符不再需要, 儘管依然允許。
拓展:
在學習arm過程中發現這“指針函數”與“函數指針”容易搞錯,所以今天,我自己想一次把它搞清楚,找了一些資料,首先它們之間的定義:
1、指針函數是指帶指針的函數,即本質是一個函數。函數返回類型是某一類型的指針
類型標識符 *函數名(參數表)
int *f(x,y);
首先它是一個函數,只不過這個函數的返回值是一個地址值。函數返回值必須用同類型的指針變量來接受,也就是説,指針函數一定有函數返回值,而且,在主調函數中,函數返回值必須賦給同類型的指針變量。
表示:
float *fun();
float *p;
p = fun(a);
注意指針函數與函數指針表示方法的不同,千萬不要混淆。最簡單的辨別方式就是看函數名前面的指針*號有沒有被括號()包含,如果被包含就是函數指針,反之則是指針函數。
來講詳細一些吧!請看下面
指針函數:
當一個函數聲明其返回值為一個指針時,實際上就是返回一個地址給調用函數,以用於需要指針或地址的表達式中。
格式:
類型説明符 * 函數名(參數)
當然了,由於返回的是一個地址,所以類型説明符一般都是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(%dn,*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];
}
程序應該是很好理解的,子函數返回的是數組某元素的地址。輸出的是這個地址裏的值。
2、函數指針是指向函數的指針變量,即本質是一個指針變量。
int (*f) (int x); /* 聲明一個函數指針 */
f=func; /* 將func函數的首地址賦給指針f */
指向函數的指針包含了函數的地址,可以通過它來調用函數。聲明格式如下:
類型説明符 (*函數名)(參數)
其實這裏不能稱為函數名,應該叫做指針的變量名。這個特殊的指針指向一個返回整型值的函數。指針的聲明筆削和它指向函數的聲明保持一致。
指針名和指針運算符外面的括號改變了默認的運算符優先級。如果沒有圓括號,就變成了一個返回整型指針的函數的原型聲明。
例如:
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(filefuncn);
}
void editfunc()
{
printf(editfuncn);
}
程序輸出為:
filefunc
editfunc
主要的區別是一個是指針變量,一個是函數。在使用是必要要搞清楚才能正確使用