定義
每一個函數都占用一段內存單元,它們有一個起始地址,指向函數入口地址的指針稱為函數指針。
語法
數據類型 (*指針變量名)(參數表):
1
|
int (*myFunc)( double b, int c); |
說明
函數指針的定義形式中的數據類型是指函數的返回值的類型。
區分下面兩個語句:
1
2
|
int (*p)( int a, int b); //p是一個指向函數的指針變量,所指函數的返回值類型為整型 int *p( int a, int b); //p是函數名,此函數的返回值類型為整型指針 |
指向函數的指針變量不是固定指向哪一個函數的,而只是表示定義了一個這樣類型的變量,它是專門用來存放函數的入口地址的;在程序中把哪一個函數的地址賦給它,它就指向哪一個函數。
在給函數指針變量賦值時,只需給出函數名,而不必給出參數。
如函數max的原型為:int max(int x, int y); 指針p的定義為:int (*p)(int a, int b); 則p = max;的作用是將函數max的入口地址賦給指針變量p。這時,p就是指向函數max的指針變量,也就是p和max都指向函數的開頭。
在一個程序中,指針變量p可以先后指向不同的函數,但一個函數不能賦給一個不一致的函數指針(即不能讓一個函數指針指向與其類型不一致的函數)。
如有如下的函數:
1
|
int fn1( int x, int y); int fn2( int x); |
定義如下的函數指針:
1
|
int (*p1)( int a, int b); int (*p2)( int a); |
則
1
2
3
|
p1 = fn1; //正確 p2 = fn2; //正確 p1 = fn2; //產生編譯錯誤 |
定義了一個函數指針并讓它指向了一個函數后,對函數的調用可以通過函數名調用,也可以通過函數指針調用(即用指向函數的指針變量調用)。
如語句:c = (*p)(a, b); //表示調用由p指向的函數(max),實參為a,b,函數調用結束后得到的函數值賦給c。
函數指針只能指向函數的入口處,而不可能指向函數中間的某一條指令。不能用*(p+1)來表示函數的下一條指令。
函數指針變量常用的用途之一是把指針作為參數傳遞到其他函數。
函數指針使用舉例
說明看代碼注釋就可以
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
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
|
#include <iostream> using namespace std; class test { public : test() { cout<< "constructor" <<endl; } int fun1( int a, char c) { cout<< "this is fun1 call:" <<a<< " " <<c<<endl; return a; } void fun2( double d) const { cout<< "this is fun2 call:" <<d<<endl; } static double fun3( char buf[]) { cout<< "this is fun3 call:" <<buf<<endl; return 3.14; } }; int main() { // 類的靜態成員函數指針和c的指針的用法相同 double (*pstatic)( char buf[]) = NULL; //不需要加類名 pstatic = test::fun3; //可以不加取地址符號 pstatic( "myclaa" ); pstatic = &test::fun3; (*pstatic)( "xyz" ); //普通成員函數 int (test::*pfun)( int , char ) = NULL; //一定要加類名 pfun = &test::fun1; //一定要加取地址符號 test mytest; (mytest.*pfun)(1, 'a' ); //調用是一定要加類的對象名和*符號 //const 函數(基本普通成員函數相同) void (test::*pconst)( double ) const = NULL; //一定要加const pconst = &test::fun2; test mytest2; (mytest2.*pconst)(3.33); // //構造函數或者析構函數的指針,貌似不可以,不知道c++標準有沒有規定不能有指向這兩者的函數指針 // (test::*pcon)() = NULL; // pcon = &test.test; // test mytest3; // (mytest3.*pcon)(); return 0; } |