一個函式指標必須用* 操作符(和一對額外的括弧) “轉換為” 一個“真正的” 函式才能呼叫:
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
主要的區別是一個是指標變數,一個是函式。在使用是必要要搞清楚才能正確使用