函數指針和指針函數用法和區別

前言
函數指針和指針函數,在學習 C 語言的時候遇到這兩個東西簡直頭疼,固然還有更頭疼的,好比什麼函數指針函數、指針函數指針、數組指針、指針數組、函數指針數組等等,描述越長其定義就越複雜,固然理解起來就越難,特別是剛開始學習這門語言的童鞋,估計碰到這些東西就已經要崩潰了,而後好不容易死記硬背下來應付考試或者面試,而後過了幾天發現,又是根本不會用,也不知道該在哪些地方用,這就尷尬了。
今天這裏只講兩個相對簡單的,其實上面說那些太複雜的東西也真的不多用,即使是用了理解起來很麻煩,因此莫不如先深入理解這兩個比較容易的,而且項目中比較經常使用到。面試

正文
先來看看二者的定義以及說明。數組

指針函數
定義
指針函數,簡單的來講,就是一個返回指針的函數,其本質是一個函數,而該函數的返回值是一個指針。
聲明格式爲:*類型標識符 函數名(參數表)函數

這彷佛並不難理解,再進一步描述一下。
看看下面這個函數聲明:學習

int fun(int x,int y);

這種函數應該都很熟悉,其實就是一個函數,而後返回值是一個 int 類型,是一個數值。
接着看下面這個函數聲明:指針

int *fun(int x,int y);
這和上面那個函數惟一的區別就是在函數名前面多了一個*號,而這個函數就是一個指針函數。其返回值是一個 int 類型的指針,是一個地址。開發

這樣描述應該很容易理解了,所謂的指針函數也沒什麼特別的,和普通函數對比不過就是其返回了一個指針(即地址值)而已。io

指針函數的寫法
int *fun(int x,int y);
int * fun(int x,int y);
int* fun(int x,int y);

這個寫法看我的習慣,其實若是*靠近返回值類型的話可能更容易理解其定義。ast

示例
(因爲本人習慣於 Qt 中進行開發,因此這裏爲了方便,示例是在 Qt 工程中寫的,其語法是同樣的,只是輸出方式不一樣)
來看一個很是簡單的示例:變量

typedef struct _Data{
    int a;
    int b;
}Data;數據類型

//指針函數
Data* f(int a,int b){
    Data * data = new Data;
    data->a = a;
    data->b = b;
    return data;
}

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    //調用指針函數
    Data * myData = f(4,5);
    qDebug() << "f(4,5) = " << myData->a << myData->b;

    return a.exec();
}

輸出以下:

f(4,5) =  4 5
注意:在調用指針函數時,須要一個同類型的指針來接收其函數的返回值。
不過也能夠將其返回值定義爲 void*類型,在調用的時候強制轉換返回值爲本身想要的類型,以下:

//指針函數
void* f(int a,int b){
    Data * data = new Data;
    data->a = a;
    data->b = b;
    return data;
}

調用:
Data * myData = static_cast<Data*>(f(4,5));
其輸出結果是同樣的,不過不建議這麼使用,由於強制轉換可能會帶來風險。

##函數指針
###定義
函數指針,其本質是一個指針變量,該指針指向這個函數。總結來講,函數指針就是指向函數的指針。
聲明格式:類型說明符 (*函數名) (參數)
以下:

int (*fun)(int x,int y);

函數指針是須要把一個函數的地址賦值給它,有兩種寫法:

fun = &Function;
fun = Function;

取地址運算符&不是必需的,由於一個函數標識符就表示了它的地址,若是是函數調用,還必須包含一個圓括號括起來的參數表。

調用函數指針的方式也有兩種:

x = (*fun)();
x = fun();

兩種方式都可,其中第二種看上去和普通的函數調用沒啥區別,若是能夠的話,建議使用第一種,由於能夠清楚的指明這是經過指針的方式來調用函數。固然,也要看我的習慣,若是理解其定義,隨便怎麼用都行啦。

示例
int add(int x,int y){
    return x+y;
}
int sub(int x,int y){
    return x-y;
}
//函數指針
int (*fun)(int x,int y);

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    //第一種寫法
    fun = add;
    qDebug() << "(*fun)(1,2) = " << (*fun)(1,2) ;
    //第二種寫法
    fun = &sub;
    qDebug() << "(*fun)(5,2) = " << (*fun)(5,3)  << fun(5,3);

    return a.exec();
}


輸出以下:

(*fun)(1,2) =  3
(*fun)(5,2) =  2 2

上面說到的幾種賦值和調用方式我都分別使用了,其輸出結果是同樣的。

兩者區別
經過以上的介紹,應該都能清楚的理解其兩者的定義。那麼簡單的總結下兩者的區別:
###定義不一樣
指針函數本質是一個函數,其返回值爲指針。
函數指針本質是一個指針,其指向一個函數。

寫法不一樣
指針函數:int* fun(int x,int y);
函數指針:int (fun)(int x,int y);
能夠簡單粗暴的理解爲,指針函數的是屬於數據類型的,而函數指針的星號是屬於函數名的。
再簡單一點,能夠這樣辨別二者:函數名帶括號的就是函數指針,不然就是指針函數。

用法不一樣
上面已經寫了詳細示例,這裏就不在囉嗦了。

總而言之,這兩個東西很容易搞混淆,必定要深刻理解其二者定義和區別,避免犯錯。  

相關文章
相關標籤/搜索