當(dāng)前位置:首頁 > 公眾號精選 > C語言與CPP編程
[導(dǎo)讀]關(guān)于指針,前面文章C語言指針詳解有過介紹,這里主要討論函數(shù)指針和指針函數(shù)。 1 什么是指針? 定義:指針是程序數(shù)據(jù)在內(nèi)存中的地址,而指針變量是用來保存這些地址的變量; 上面一個 4GB 的內(nèi)存可以存放 2^32 字節(jié)的數(shù)據(jù)。左側(cè)連續(xù)的十六進(jìn)制編號就是內(nèi)存地址

關(guān)于指針,前面文章C語言指針詳解有過介紹,這里主要討論函數(shù)指針和指針函數(shù)。

1 什么是指針?

定義:指針是程序數(shù)據(jù)在內(nèi)存中的地址,而指針變量是用來保存這些地址的變量;


上面一個 4GB的內(nèi)存可以存放 2^32字節(jié)的數(shù)據(jù)。左側(cè)連續(xù)的十六進(jìn)制編號就是內(nèi)存地址,每個內(nèi)存地址對應(yīng)一個字節(jié)的內(nèi)存空間。而指針變量保存的就是這個編號,也即內(nèi)存地址。

指針的聲明:

指針其實就是一個變量,指針的聲明方式與一般的變量聲明類似,如下:

int *p;         // 聲明一個 int 類型的指針 p,該指針指向一個int類型的對象
char *p         // 聲明一個 char 類型的指針 p,該指針指向一個int類型的對象
int *arr[10]    // 聲明一個指針數(shù)組,該數(shù)組有10個元素,其中每個元素都是一個指向 int 類型對象的指針
int (*arr)[10]  // 聲明一個數(shù)組指針,該指針指向一個 int 類型的一維數(shù)組
int **p;        // 聲明一個指針 p ,該指針指向一個 int 類型的指針

聲明一個指針變量并不會自動分配任何內(nèi)存。在對指針進(jìn)行間接訪問之前,指針必須進(jìn)行初始化:或是使他指向現(xiàn)有的內(nèi)存,或者給他動態(tài)分配內(nèi)存,否則我們并不知道指針指向哪兒,這個問題需要特別關(guān)注。

2 什么是函數(shù)指針?

函數(shù)指針定義:函數(shù)指針是指向函數(shù)的指針變量。因此“函數(shù)指針”本身首先應(yīng)是指針變量,只不過該指針變量指向函數(shù)。這正如用指針變量可指向整型變量、字符型、數(shù)組一樣,這里是指向函數(shù)。

其通用表達(dá)式為:類型說明符 (*函數(shù)名) (參數(shù))

int (*fun)(int x)  //函數(shù)指針的定義
int (*fun)(int x,int y) //函數(shù)指針的定義

函數(shù)指針在PC軟件開發(fā)中使用較少,在嵌入式行業(yè)使用較多,但是無論是PC軟件還是嵌入式軟件,理解函數(shù)指針的定義和使用,對于理解程序設(shè)計都是很有好處的。

函數(shù)指針的賦值

函數(shù)指針和其他指針一樣定義之后使用之前也是需要初始化。

函數(shù)指針有兩個用途:調(diào)用函數(shù)和做函數(shù)的參數(shù)

int (*fun)(int x,int y) //函數(shù)指針的定義
fun = &Function          //函數(shù)指針的賦值方式1
fun = Function           //函數(shù)指針的賦值方式2
x = (*fun)()             //函數(shù)指針的調(diào)用方式1
x = fun()                //函數(shù)指針的調(diào)用方式2

函數(shù)賦值的時候取地址運(yùn)算符&不是必需的,因為一個函數(shù)標(biāo)識符就表示了它的地址,并且賦值的時候函數(shù)不需要帶圓括號;

如果是函數(shù)調(diào)用,還必須包含一個圓括號括起來的參數(shù)表。

函數(shù)指針的用法

我們使用指針的時候,需要通過鑰匙 *來取其指向的內(nèi)存里面的值,函數(shù)指針使用也如此。通過用(*pf)取出存在這個地址上的函數(shù),然后調(diào)用它。

char*  fun(char* p1,char* p2)
{
  int i = 0;
  i = strcmp(p1,p2); if(0 == i)
  { return p1;
  } else { return p2;
  }
}
int main()
{
  char * (*pf)(char* p1,char* p2);
  pf = &fun;
  (*pf)("aa","bb"); return 0;
}

這里需要注意到是,在Visual C++6.0里,給函數(shù)指針賦值時,可以用&fun或直接用函數(shù)名fun。這是因為函數(shù)名被編譯之后其實就是一個地址,所以這里兩種用法沒有本質(zhì)的差別。

用法延申

當(dāng)我們不滿足于函數(shù)指針上面如此簡單的用法時,這時候需要一個高級用法來擴(kuò)展我們對于函數(shù)指針的認(rèn)知邊界。

感興趣的同學(xué)可以看看下面這個用法,并嘗試?yán)斫庠摫磉_(dá)式是如何使用的函數(shù)指針。

(* (void(*)()) 0)(); //出自《C Trap and Pitfalls》這本經(jīng)典的書

答案如下:   ``

  • 第一步:通過 void(*) (),可以明白這是一個函數(shù)指針類型。這個函數(shù)沒有參數(shù),沒有返回值。
  • 第二步:通過 (void(*) ())0,可以明白這是將 0強(qiáng)制轉(zhuǎn)換為函數(shù)指針類型, 0是一個地址,也就是說一個函數(shù)存在首地址為 0的一段區(qū)域內(nèi)。
  • 第三步:通過 (*(void(*) ())0),可以明白這是取0地址開始的一段內(nèi)存里面的內(nèi)容。
  • 第四步:最終理解 (*(void(*) ())0)(),這是函數(shù)調(diào)用。

讓程序跳轉(zhuǎn)到絕對地址為0x0113F90C

方法一:

  • 0x0113F90C地址強(qiáng)制轉(zhuǎn)換為函數(shù)指針類型,即: (void (*)())0x0113F90C
  • 然后調(diào)用: ((void (*)())0x0113F90C)()

方法二:

typedef (void (*)())  VoidFuncPtr;
((VoidFuncPtr)0x0113F90C)();

面試題:指出程序的錯誤

#include void main(void)
{
   int max(x,y);
   int *p=max;
   int a,b,c,d;
   scanf("%d %d %d",a,b,c);
   d=p(p(a,b),c); printf("d:%d\n",d); return;
}
int max(int a,int y)
{ return(x > y ?x:y);
}

答案

  • int max(x ,y);函數(shù)聲明錯誤,改為: int max(int x,int y);

解析:max函數(shù)聲明只是寫出了函數(shù)的形參的名稱,這對參數(shù)的類型來說是毫無意義的,編譯器會把x和y當(dāng)做數(shù)據(jù)類型來看,編譯時會出錯,max的調(diào)用肯定也會出錯。

  • int *p=max;指針定義錯誤,改為: int (*p)(int ,int)=max;

解析:函數(shù)的指針是不能直接賦值給int型指針.

  • scanf("%d %d %d",a,b,c);庫函數(shù)使用錯誤,改為 scanf("%d %d%d",&a,&b,&c);

解析:庫函數(shù)使用錯誤,第二部分應(yīng)該是接收數(shù)據(jù)的地址,這里卻寫成了變量。

  • d=p(p(a,b),c);函數(shù)指針調(diào)用函數(shù)錯誤,改為 d=(*p)((*p)(a,b),c);`

解析:用函數(shù)指針調(diào)用函數(shù)的格式如下:(【*】【函數(shù)指針名稱】)(【參數(shù)列表】);不能直接用函數(shù)指針加上參數(shù)就直接調(diào)用

3 什么是指針函數(shù)?

指針函數(shù)定義:指針函數(shù)的落腳點是一個函數(shù),這個函數(shù)的返回值是一個指針,與普通函數(shù)int function(int,int)類似,只是返回的數(shù)據(jù)類型不一樣而已。

_type_ *function(int, int) //返回的是指針地址
int function(int,int)     //返回的是int型數(shù)據(jù)。
int  *fun(int x)        //指針函數(shù)的定義
int * fun(int x,int y)  //指針函數(shù)的定義
int* fun(int x,int y)   //指針函數(shù)的定義

以上三種寫法均正確,但是*靠近返回值一點更容易理解。

指針函數(shù)的調(diào)用

在調(diào)用指針函數(shù)時,需要一個同類型的指針來接收其函數(shù)的返回值。

typedef struct _Data{
    int a;
    int b;
}Data;

//指針函數(shù)
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);
    //調(diào)用指針函數(shù)
    Data * myData = f(4,5);
    qDebug() << "f(4,5) = " << myData->a << myData->b; return a.exec();
}

不過也可以將其返回值定義為 void*類型,在調(diào)用的時候強(qiáng)制轉(zhuǎn)換返回值為自己想要的類型。

其輸出結(jié)果是一樣的,不過不建議這么使用,因為強(qiáng)制轉(zhuǎn)換可能會帶來風(fēng)險。返回類型可以是任何基本類型和復(fù)合類型。返回指針的函數(shù)的用途十分廣泛。

事實上,每一個函數(shù),即使它不帶有返回某種類型的指針,它本身都有一個入口地址,該地址相當(dāng)于一個指針。

比如函數(shù)返回一個整型值,實際上也相當(dāng)于返回一個指針變量的值,不過這時的變量是函數(shù)本身而已,而整個函數(shù)相當(dāng)于一個“變量”。

4 函數(shù)指針與指針函數(shù)區(qū)別

通過以上的介紹,小伙伴應(yīng)該都能理解二者的定義。那么簡單的總結(jié)下二者的區(qū)別:

1. 定義不同

  • 指針函數(shù)本質(zhì)是一個函數(shù),其返回值為指針。
  • 函數(shù)指針本質(zhì)是一個指針,其指向一個函數(shù)。

2. 寫法不同

  • 指針函數(shù): int* fun(int x,int y);
  • 函數(shù)指針: int (*fun)(int x,int y);

可以簡單粗暴的理解為,指針函數(shù)的*是屬于數(shù)據(jù)類型的,而函數(shù)指針的星號是屬于函數(shù)名的。

再簡單一點,可以這樣辨別兩者:函數(shù)名帶括號的就是函數(shù)指針,否則就是指針函數(shù)。

3. 用法不同

上面函數(shù)指針和指針函數(shù)的用法都有,但是函數(shù)指針的用法會更多,相對而言難度也更大,例如函數(shù)指針與回調(diào)函數(shù),如果是C++非靜態(tài)成員函數(shù)指針,其用法也會有一些區(qū)別,感興趣的同學(xué)可以關(guān)注后續(xù)推文或自行查閱相關(guān)書籍。

總而言之,這兩個東西很容易搞混淆,一定要深入理解其兩者定義和區(qū)別,避免犯錯。


免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉