1. 程式人生 > >c語言函式指標學習心得

c語言函式指標學習心得

顧名思義,指標函式即返回指標的函式。其一般定義形式如下:

型別名*函式名(函式引數表列);

其中,字尾運算子括號“()表示這是一個函式,其字首運算子星號“*”表示此函式為指標型函式,其函式值為指標,即它帶回來的值的型別為指標,當呼叫這個函式後,將得到一個“指向返回值為的指標(地址),“型別名”表示函式返回的指標指向的型別”。

(函式引數表列)中的括號為函式呼叫運算子,在呼叫語句中,即使函式不帶引數,其引數表的一對括號也不能省略。其示例如下:

int*pfun(int, int);

由於“*”的優先順序低於“()”的優先順序,因而pfun首先和後面的“()”結合,也就意味著,pfun是一個函式。即:

int*(pfun(int, int));

接著再和前面的“*”結合,說明這個函式的返回值是一個指標。由於前面還有一個int,也就是說,pfun是一個返回值為整型指標的函式。

我們不妨來再看一看,指標函式與函式指標有什麼區別?

int (*pfun)(int, int);

通過括號強行將pfun首先與“*”結合,也就意味著,pfun是一個指標,接著與後面的“()”結合,說明該指標指向的是一個函式,然後再與前面的int結合,也就是說,該函式的返回值是int。由此可見,pfun是一個指向返回值為int的函式的指標。

雖然它們只有一個括號的差別但是表示的意義卻截然不同。函式指標的本身是一個指標,指標指向的是一個函式。指標函式的本身是一個函式,其函式的返回值是一個指標。

在上面提到的指標函式裡面,有這樣一類函式,它們也返回指標型資料(地址),但是這個指標不是指向intchar之類的基本型別,而是指向函式。對於初學者,別說寫出這樣的函式宣告,就是看到這樣的寫法也是一頭霧水。比如,下面的語句:

int(*ff(int))(int *, int);

我們用上面介紹的方法分析一下,ff首先與後面的()結合,即:

int(*(ff(int)))(int *, int);//用括號將ff(int)再括起來

也就意味著ff是一個函式。

接著與前面的*結合,說明ff函式的返回值是一個指標。然後再與後面的()結合,也就是說,該指標指向的是一個函式。

這種寫法確實讓人非常難懂,以至於一些初學者產生誤解,認為寫出別人看不懂的程式碼才能顯示自己水平高。而事實上恰好相反,能否寫出通俗易懂的程式碼是衡量程式設計師是否優秀的標準。一般來說,用

typedef關鍵字會使該宣告更簡單易懂。在前面我們已經見過:

int(*PF)(int *, int);

也就是說,PF是一個函式指標“變數”。當使用typedef聲明後,則PF就成為了一個函式指標“型別”,即:

typedefint (*PF)(int *, int);

這樣就定義了返回值的型別。然後,再用PF作為返回值來宣告函式:

PF ff(int);

下面是一個例項:

1 #include<stdio.h>
  2 typedef void (*FUNC)(int a);
  3
  4 void func (int a)
  5 {
  6         printf("func0\n");
  7 }
  8
  9 void func1 (int b)
 10 {
 11         printf("func1\n");
 12 }
 13
 14 void func2 (int c)
 15 {
 16         printf("func2\n");
 17 }
 18
 19 void func3 (int c)
 20 {
 21         printf("sorry\n");
 22 }
 23
 24
 25 FUNC func4 (int c,FUNC temp)
 26 {
 27         printf("func4\n");
 28         (*temp)(0);
 29         return func2;
 30 }
 31
 32 FUNC my_func(int c)
 33 {
 34         switch(c)
 35         {
 36                 case 0:return func ;break;
 37                 case 1:return func1 ;break;
 38                 case 2:return func2;break;
 39                 default:return func3;break;
 40         }
 41 }
           int main(void)
 44 {
 45         my_func(3)(5);
 46         func4(1,func1)(0);
 47         return 0;
 48 }
 49

over!!