在學習arm過程中發現這“指針函數”與“函數指針”容易搞錯,最簡單的辨別方式就是看函數名前面的指針*號有沒有被括號()包含,如果被包含就是函數指針,反之則是指針函數。
今天我們從頭把它搞清楚
首先它們之間的定義:
1、指針函數是指帶指針的函數,即本質是一個函數,函數返回類型是某一類型的指針。
類型標識符 *函數名(參數表)
int *f(x,y);
首先它是一個函數,只不過這個函數的返回值是一個地址值。函數返回值必須用同類型的指針變量來接受,也就是說,指針函數一定有函數返回值,而且,在主調函數中,函數返回值必須賦給同類型的指針變量。
表示:
float *fun(); float *p; p = fun(a);
來講詳細一些吧!請看下面
注意指針函數與函數指針表示方法的不同,千萬不要混淆。
指針函數:
當一個函數聲明其返回值為一個指針時,實際上就是返回一個地址給調用函數,以用於需要指針或地址的表達式中。
格式:
類型說明符 * 函數名(參數)
當然了,由於返回的是一個地址,所以類型說明符一般都是int。
例如:
int *GetDate(); int * aaa(int,int);
函數返回的是一個地址值,經常使用在返回數組的某一元素地址上。
1 int * GetDate(int wk,int dy); 2 main() 3 { 4 int wk,dy; 5 do{ 6 printf(Enter week(1-5)day(1-7)\n); 7 scanf(%d%d,&wk,&dy); 8 } 9 while(wk<1||wk>5||dy<1||dy>7); 10 printf(%d\n,*GetDate(wk,dy)); 11 } 12 13 int * GetDate(int wk,int dy) 14 { 15 static int calendar[5][7]= 16 { 17 {1,2,3,4,5,6,7}, 18 {8,9,10,11,12,13,14}, 19 {15,16,17,18,19,20,21}, 20 {22,23,24,25,26,27,28}, 21 {29,30,31,-1} 22 }; 23 return &calendar[wk-1][dy-1]; 24 }
程序應該是很好理解的,子函數返回的是數組某元素的地址。輸出的是這個地址里的值。
2、函數指針是指向函數的指針變量,即本質是一個指針變量。
int (*f) (int x); /*聲明一個函數指針 */ f=func; /* 將func函數的首地址賦給指針f */
指向函數的指針包含了函數的地址的入口地址,可以通過它來調用函數。聲明格式如下:
類型說明符 (*函數名) (參數)
其實這里不能稱為函數名,應該叫做指針的變量名。這個特殊的指針指向一個返回整型值的函數。指針的聲明筆削和它指向函數的聲明保持一致。
指針名和指針運算符外面的括號改變了默認的運算符優先級。如果沒有圓括號,就變成了一個返回整型指針的函數的原型聲明。
例如:
void (*fptr)();
把函數的地址賦值給函數指針,可以采用下面兩種形式:
fptr=&Function;
fptr=Function;
取地址運算符&不是必需的,因為單單一個函數標識符就標號表示了它的地址,如果是函數調用,還必須包含一個圓括號括起來的參數表。
可以采用如下兩種方式來通過指針調用函數:
x=(*fptr)();
x=fptr();
第二種格式看上去和函數調用無異。但是有些程序員傾向於使用第一種格式,因為它明確指出是通過指針而非函數名來調用函數的。
下面舉一個例子:
1 void (*funcp)(); 2 void FileFunc(),EditFunc(); 3 4 main() 5 { 6 funcp=FileFunc; 7 (*funcp)(); 8 funcp=EditFunc; 9 (*funcp)(); 10 } 11 12 void FileFunc() 13 { 14 printf(FileFunc\n); 15 } 16 17 void EditFunc() 18 { 19 printf(EditFunc\n); 20 }
程序輸出為:
FileFunc
EditFunc
主要的區別是一個是指針變量,一個是函數。在使用是必要要搞清楚才能正確使用
轉自:http://www.cnblogs.com/gmh915/archive/2010/06/11/1756067.html
有整理與刪改