C++函數指針詳解

jopen 9年前發布 | 14K 次閱讀 C/C++開發 C/C++

學習c++的過程中,指針是難點,熟悉了指針之后,還有一個讓人很蛋疼的難點,那就是函數指針了。本博文詳細介紹一下常見的各種坑爹的函數指針。

至于指針的詳細學習,推薦這篇博文C++指針詳解

與數據一樣,函數也有地址,函數的地址就是內存中存放函數語言代碼的起始地址。函數指針就是指向這個地址。函數指針所指向的類型,就是函數本身。我 們知道,指針所指向類型代表了指針所指向的內存區域的大小。所以函數指針所指向的類型,就是函數在內存中所占據內存的大小。知道了函數的起始地址和大小, 所以函數指針可以很輕易的代替函數完成函數調用。

一、最簡單的函數指針

變量都包括聲明和賦值,指針不例外,函數指針也不例外。我們來看一個簡單的函數:

void add(int a, int b){
    cout << a + b << endl;
}
</div>

一個簡單的加法計算并輸出到命令行的函數。

那么如何通過函數指針來調用它呢?

1、聲明:

void (*p1)(int a, int b);
</div>

函數指針的聲明很簡單,基本就是通過一個指針把函數名替換。指針p1的類型為void (*) (int a,int b),表明指針是一個指向某個函數的指針,指針指向的類型為void () (int a,int b)

2、賦值:

p1 = add;
</div>

3、也可以直接定義:

void (*p1)(int a, int b) = add;
</div>

注意,函數void add(int a,int b)的函數名add就是函數的地址。將地址add賦值給指針p1,那么就可以通過函數指針p1直接調用函數了。

4、調用:

(*p1)(1, 2);
p1(1, 2);
</div>

注意!出于歷史原因以上2種方式都是可以調用函數的。

二、包含多個函數指針的數組

有時候有這種情況,有一個數組,數組中的每個元素都是一個函數指針,該怎么定義這個數組呢?

1、解釋*p[n]和(*p)[n]

我們知道,[]運算符的優先級要高于*,所以,p[3]表示含有3個元素的數組,而*p[3] 前面的 ” * ” 指明了數組中元素的類型,即*p[3]表示一個指向3個指針的數組。

p[3]表示含有3個元素的數組,那么(*p)[3]就是用 *p 替換了 p,很容易想到,(*p)[3] 表示指向一個包含3個元素的數組的指針。

2、聲明:

void (*p2[2])(int a, int b);
</div>

數組名為p2,數組大小為2,數組中元素類型為void (*)(int a, int b),表明元素是一個指向某個函數的指針,指針指向的類型為void () (int a,int b)。

3、賦值:

p2[1] = add;
</div>

理解上跟上面是一樣的。

4、調用:

p2[1](2,3);
(*p2[1])(3,4);
</div>

同樣是2種方式都可以。

三、指向“包含多個函數指針的數組“的指針

這個標題好像有點拗口。簡而言之,這個指針指向上文中的 “包含多個函數指針的數組” 。其實很簡單,說白了,就是把上文中的p2用一個指針來代替。

1、聲明:

void (*(*p3)[2])(int a, int b);
</div>

可以看到,無非就是把p2用*p3代替。

2、賦值,注意,既然是指針,使用前必須初始化:

p3 = &p2;
(*p3)[1] = add;
</div>

注意!既然實質上就是把p2用*p3代替,c++11可以很簡單的這樣直接定義:auto p3 = &p2; 代替了void (*(*p3)[2])(int a, int b)= &p2;

3、調用:

(*p3)[1](1, 2);
((*p3)[1])(1, 2);
</div> 來源:不只是程序員

 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!