c++函數名指針和函數指針

前言

我們先來看一下函數指針式如何定義的,假如我們有一個函數int fun(int){…};那麼他對應的函數指針寫法就應該是int (*p)(int);然後再對他進行賦值,即p=fun;之後你就可以在接下來的地方按p作為函數名來調用它用起來完全和fun一樣。(註意這裡的p指針並不是隻能接受fun這個函數名,任何返回值是int,參數隻有一個int的函數都可以把函數名賦給p)

首先說一下C/C++在創建一個變量的時候比如int a;相應的在內存就會分配一個4個字節(根據不同機器可能不同)空間來存放這個int變量,而假設這4個字節的起始地址是0XFF0A,那麼實際上就存在一種變量名和內存地址的映射,即a可以看做是一個標示符,他隻是代表著0XFF0A這個地址,在程序中你對a進行的操作實際上也就是對內存中以0XFF0A為首地址的4個字節的操作,特別是如果對a進行取地址操作也就是&a實際上就是返回0XFF0A這個地址值,實際上你可以看成就是返回一個指向這個地址的指針(如果你覺的不能理解,就當我沒說吧).同理對於我們在程序中創建的函數,他是保存在程序中的單獨區域的,而我們調用它們就像使用變量一樣需要一個地址來唯一的指向它,所以每個函數都需要一個地址來唯一標識自己(也就是我們常說的入口地址),就像上面的a對應0XFF0A,那麼假設我們定義瞭一個int fun(int){};函數的入口地址是0XAAEE,則fun也就是函數名他會映射0XAAEE,和上面的int變量a一樣如果對它進行取地址&fun的話就會返回0XAAEE,實際上fun也是一種類型,就當它是函數名類型好瞭,隻要記住函數名本身並不是一個指針類型就可以瞭。

在調用函數的時候有函數名就夠瞭,比如fun(2);不要以為隻要有函數名就能調用函數瞭,其實這隻是寫法上的一個迷惑點,而編譯器在編譯的時候一律都會進行所謂的"Function-to-pointer conversion",也就是把函數名隱式轉換成函數指針類型,也就是要通過函數指針來調用函數,所以如果你在調用函數的時候寫成(&fun)(2)也是一樣能工作的,因為&fun實際上就是返回一個函數指針,參照上一段中&a的例子,隻是這種寫法很不常見,即使你不顯式的寫出&的話編譯器也會隱式的進行轉換,註意&fun左右的括號必須有,這是因為運算符優先級的問題。

其實即使寫成(fun)(2)也是可以正常運行的,這是因為當編譯器看到fun的時候發現它前面沒有&也就是沒有給他顯示的轉換成指針那麼他就要隱式的轉換成指針,當轉換完之後發現前面又有一個這時候也就是要進行所謂的"解引用"操作,也就是到*後面裡指針裡取出來值,而那麼值實際上也就也就是0XAAEE也就是函數名fun,這麼一次隱式換然後再來一次解引用實際上相當於什麼也沒做,所以系統還會再進行一次隱式的"Function-to-pointer conversion",即使你寫成(*******fun)(2)也會正常運行,和剛才的一個道理,隻是多做瞭幾次反復的轉解操作而已,都是編譯器自己完成的,不必去理會!

例 1

#include<iostream>
using namespace std;
void fun(int a)
{

}

int main()
{
    cout<<fun<<endl;
    cout<<*fun<<endl;
    cout<<&fun<<endl;
    cout<<*****fun<<endl;
}

結果輸出的值都是一樣的,也就是都是指向同一個函數地址的指針值。

例 2

下面再結合自己定義的函數指針來看看:

#include<iostream>
using namespace std;
int fun(int a)
{
    cout<<"fun"<<endl;
    return 0;
}

void main()
{
    int(*p)(int)=fun;
    int(*p1)(int)=*fun;
    int(*p2)(int)=&fun;
    p(1);
    p1(1);
    p2(1);
}

例 3

發現函數都能正常的運行,其實p1,p2,p和fun賦值之後大傢一樣理解就行瞭。

代碼:

#include<iostream>
using namespace std;
int fun(int a)
{
    cout<<"fun"<<endl;
    return 0;
}
 
void main()
{
    int(*p)(int)=fun;
    p(1);
//  (&p)(1);   
    (*p)(1);
    (****p)(1);
}

上面的程序也都會正常的運行,隻要再理解的時候把p當成隻是對函數名多做瞭一次轉換就可以瞭,接下來理解都一樣!註意上面註釋掉的哪一行是不能運行的,因為p是我們自己定義的函數指針類型,如果你對指針取地址那麼將得到p這個變量本身的地址,這就不能正確調用函數瞭!再多說一句,其實你如果運行&&fun這個式子也是非法的,至於為什麼,大傢一起幫我思考思考,我個人認為當我們運行&fun的時候他會轉換成函數指針而實際上這個指針隻是一個臨時值而臨時值是沒有實際存放的內存地址的所以也就無法繼續取地址瞭!

到此這篇關於c++函數名指針和函數指針的文章就介紹到這瞭,更多相關c++函數名指針和函數指針內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: