函數指針大家了解一下就行了,用得不多,但一定要認識它。
什么是函數指針
如果在程序中定義了一個函數,那么在編譯時系統就會為這個函數代碼分配一段存儲空間,這段存儲空間的首地址稱為這個函數的地址。而且函數名表示的就是這個地址。既然是地址我們就可以定義一個指針變量來存放,這個指針變量就叫作函數指針變量,簡稱函數指針。
那么這個指針變量怎么定義呢?雖然同樣是指向一個地址,但指向函數的指針變量同我們之前講的指向變量的指針變量的定義方式是不同的。例如:
int(*p)(int, int);
這個語句就定義了一個指向函數的指針變量 p。首先它是一個指針變量,所以要有一個“*”,即(*p);其次前面的 int 表示這個指針變量可以指向返回值類型為 int 型的函數;后面括號中的兩個 int 表示這個指針變量可以指向有兩個參數且都是 int 型的函數。所以合起來這個語句的意思就是:定義了一個指針變量 p,該指針變量可以指向返回值類型為 int 型,且有兩個整型參數的函數。p 的類型為 int(*)(int,int)。
所以函數指針的定義方式為:
函數返回值類型 (* 指針變量名) (函數參數列表);
“函數返回值類型”表示該指針變量可以指向具有什么返回值類型的函數;“函數參數列表”表示該指針變量可以指向具有什么參數列表的函數。這個參數列表中只需要寫函數的參數類型即可。
我們看到,函數指針的定義就是將“函數聲明”中的“函數名”改成“(*指針變量名)”。但是這里需要注意的是:“(*指針變量名)”兩端的括號不能省略,括號改變了運算符的優先級。如果省略了括號,就不是定義函數指針而是一個函數聲明了,即聲明了一個返回值類型為指針型的函數。
那么怎么判斷一個指針變量是指向變量的指針變量還是指向函數的指針變量呢?首先看變量名前面有沒有“*”,如果有“*”說明是指針變量;其次看變量名的后面有沒有帶有形參類型的圓括號,如果有就是指向函數的指針變量,即函數指針,如果沒有就是指向變量的指針變量。
最后需要注意的是,指向函數的指針變量沒有 ++ 和 -- 運算。
如何用函數指針調用函數
給大家舉一個例子:
int Func(int x); /*聲明一個函數*/ int (*p) (int x); /*定義一個函數指針*/ p = Func; /*將Func函數的首地址賦給指針變量p*/
賦值時函數 Func 不帶括號,也不帶參數。由於函數名 Func 代表函數的首地址,因此經過賦值以后,指針變量 p 就指向函數 Func() 代碼的首地址了。
下面來寫一個程序,看了這個程序你們就明白函數指針怎么使用了:
# include <stdio.h> int Max(int, int); //函數聲明 int main(void) { int(*p)(int, int); //定義一個函數指針 int a, b, c; p = Max; //把函數Max賦給指針變量p, 使p指向Max函數 printf("please enter a and b:"); scanf("%d%d", &a, &b); c = (*p)(a, b); //通過函數指針調用Max函數 printf("a = %d\nb = %d\nmax = %d\n", a, b, c); return 0; } int Max(int x, int y) //定義Max函數 { int z; if (x > y) { z = x; } else { z = y; } return z; }
輸出結果是:
please enter a and b:3 4
a = 3
b = 4
max = 4
