欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++ 函數(shù)指針詳細(xì)總結(jié)

 更新時間:2021年11月16日 11:11:25   作者:梁唐  
這篇文章主要介紹了C++ 函數(shù)指針內(nèi)容,下面文章圍繞C++ 函數(shù)指針的相關(guān)資料展開詳細(xì)內(nèi)容,包括函數(shù)指針的進(jìn)階內(nèi)容,需要的朋友可以參考一下,希望對大家有所幫助

1、函數(shù)指針

函數(shù)指針顧名思義,就是指向函數(shù)的指針。

和數(shù)據(jù)類似,C++當(dāng)中函數(shù)也有地址,函數(shù)的地址是存儲函數(shù)機(jī)器語言代碼的內(nèi)存地址。我們可以將另外一個函數(shù)的地址作為參數(shù)傳入函數(shù),從而實現(xiàn)函數(shù)的靈活調(diào)用。

1.1 獲取函數(shù)地址

獲取函數(shù)地址的方法非常簡單,只要使用函數(shù)名(后面不跟參數(shù)和括號)即可。比如我們有一個函數(shù)叫做think,那么think()是調(diào)用函數(shù)拿到結(jié)果,而think則是獲取函數(shù)的地址。

如果我們想要將think函數(shù)當(dāng)做參數(shù)傳入另外一個函數(shù),我們可以這么寫:

func(think);

1.2 聲明函數(shù)指針

聲明函數(shù)指針和聲明函數(shù)類似,我們聲明一個函數(shù)可以這么寫:

double process(int);


而我們聲明函數(shù)指針則可以寫成這樣:

double (*pt)(int);


如果我們把(*pt)替換成函數(shù)名的話,這其實就是一個函數(shù)的聲明。如果(*pt)是函數(shù)的話,那么pt自然就是指向函數(shù)的指針了。

1.3 函數(shù)指針傳參

如果我們要實現(xiàn)一個函數(shù),它的一個參數(shù)是一個函數(shù)指針,它的寫法和剛才一樣:

double func(double x, double (*pt)(int));


在這個聲明當(dāng)中,它的第二個參數(shù)是一個函數(shù)指針。指向的函數(shù)接收一個int參數(shù),返回一個double結(jié)果。

1.4 調(diào)用函數(shù)

最后, 我們來看下通過指針調(diào)用函數(shù)的部分。

其實也非常簡單,因為我們前面說了(*pt)的效果和函數(shù)是一樣的,我們之前通過函數(shù)名調(diào)用函數(shù),那么我們只需要改成通過(*pt)調(diào)用即可。

如:

double process(int);
double (*pt)(int);

pt = process;
cout << (*pt)(5) << endl;

2、函數(shù)指針進(jìn)階

簡單的函數(shù)指針比較簡單,但對于復(fù)雜的情況則顯得有些恐怖。下面我們來看下C++ primer當(dāng)中提供的一些例子:

const double* f1(const double ar[], int n);
const double* f2(const double [], int);
const double* f3(const double *, int);

這三個函數(shù)看起來長得不一樣,但其實是等價的。因為在函數(shù)參數(shù)列表當(dāng)中,數(shù)組和指針是等價的。其次我們可以在函數(shù)的原型中省略掉變量名,因此const double ar[]可以簡化成const double [] ,也可以寫成const double *。

有了這三個函數(shù)之后,假設(shè)我們要聲明一個指針,指向這三個函數(shù)。根據(jù)我們前文當(dāng)中說過的,可以將函數(shù)名替換成(*pt)來實現(xiàn):

const double* (*pt)(const double *, int) = f1;

其實這個語句看起來就有些復(fù)雜了,整個語句的可讀性很差。如果不是知道這里用的是一個函數(shù)指針,乍一看想要看明白估計不太容易。我們可以分成兩個部分來理解,其中const double *是一個整體,表示函數(shù)的返回值類型是一個const double *也就是一個常量浮點數(shù)的地址。其次(*pt)是一個整體,代替了函數(shù)名,表示這是一個指向函數(shù)的指針。

在C++11當(dāng)中提供了叫做auto的新特性,它可以幫助變量自動識別對應(yīng)的類型,可以解決一些類型特別復(fù)雜的問題,比如:

auto p2 = f2;

函數(shù)指針有兩種調(diào)用方法,除了可以使用(*p2)的方式調(diào)用之外,

也可以直接使用名稱調(diào)用:

const double* x = p2(ar, 3);
const double* y = (*p2)(ar, 3);

顯然前者更好,更清楚。這里其實有一個疑問,為什么這兩種方式都可以執(zhí)行呢?這是因為當(dāng)我們執(zhí)行auto p2 = f2的時候,其實是執(zhí)行的auto p2 = &f2 ,C++會隱式地將函數(shù)轉(zhuǎn)換成函數(shù)的地址。因為函數(shù)的值本身就是一個地址,所以這兩種方式才都能正確地運行。

問題還沒有結(jié)束,假如我們要定義一個指向函數(shù)的指針數(shù)組呢?這應(yīng)該怎么聲明?

也就是const double* (*pt)(const double *, int)這樣一個類型的數(shù)組,它應(yīng)該怎么聲明,這個方括號應(yīng)該放在那里?

正確答案是放在括號里:

const double* (*pt[3])(const double *, int);

因為運算符[]的優(yōu)先級高于*,因此*pt[3]表示pt是一個長度為3的指針數(shù)組。其他的內(nèi)容表明了該指針的類型。

由于我們定義的是一個數(shù)組,所以這里不能使用auto,因為自動類型推斷只能用于單值初始化而不能用于初始化列表。

到這里還沒結(jié)束,還有更恐怖的,如果我們想要定義一個指向這個數(shù)組的指針,應(yīng)該怎么辦呢?如果使用auto可以寫成:

auto ptr = &pt;

如果不使用auto呢?首先我們可以想到,這個聲明是基于pt的,我們需要在pt的聲明上加上一個*,但問題是加在哪里呢?

進(jìn)一步分析,會發(fā)現(xiàn)我們需要指出這是一個指針,而不是數(shù)組。意味著核心的部分應(yīng)該寫成(*ptr)[3],表示這是一個指向長度為3的數(shù)組的指針。因為[]的優(yōu)先級更高,所以需要使用括號。如果寫成*ptr[3]表示這是長度為3的指針數(shù)組。

我們進(jìn)一步倒推,(*ptr)[3]這個數(shù)組當(dāng)中的元素是什么類型呢?是指向函數(shù)的指針,所以寫出來結(jié)果是這樣:

const double *(*(*ptr)[3])(const double*, int) = &pt;

很明顯,這樣的定義非常非常的難以理解。而且這還不是最復(fù)雜的情況,比如函數(shù)的返回類型又是一個指向一個函數(shù)的指針……明擺著告訴我們含義我們?nèi)匀灰魄靡粫?,如果在一段不明的代碼當(dāng)中遇到,可能會直接抓狂吧……

也正因此,C++11當(dāng)中推出了auto特性,可以簡化這種情況。

多說一句題外話,golang語言當(dāng)中將變量的類型放在變量的后面而不是前面,其中一個原因就是為了解決類似情況的復(fù)雜性。

如果是golang來定義同樣的內(nèi)容,會是這樣的:

func f2(arr []float64, n int) *float64 {
    // todo
}

// 函數(shù)指針
var p1 func([]float64, int) *float64 = f2;
// 函數(shù)指針數(shù)組
var pt [3]func([]float64, int) *float64;
// 函數(shù)指針數(shù)組的指針
var ptr *[3]func([]float64, int) *float64 = &pt;

很明顯,雖然變量類型寫在變量后面剛開始會不太習(xí)慣,但是很明顯這樣要清晰很多。

到此這篇關(guān)于C++ 函數(shù)指針內(nèi)容總結(jié)的文章就介紹到這了,更多相關(guān)C++ 函數(shù)指針內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

文章轉(zhuǎn)自公眾號:Coder梁(ID:Coder_LT)

相關(guān)文章

  • C語言詳細(xì)分析講解內(nèi)存管理malloc realloc free calloc函數(shù)的使用

    C語言詳細(xì)分析講解內(nèi)存管理malloc realloc free calloc函數(shù)的使用

    C語言內(nèi)存管理相關(guān)的函數(shù)主要有realloc、calloc、malloc、free等,下面這篇文章主要給大家介紹了關(guān)于C語言內(nèi)存管理realloc、calloc、malloc、free函數(shù)的相關(guān)資料,需要的朋友可以參考下
    2022-05-05
  • 關(guān)于C++地址交換的實現(xiàn)

    關(guān)于C++地址交換的實現(xiàn)

    在C++中,地址交換通常是指通過指針操作改變兩個變量之間的內(nèi)存地址引用,當(dāng)你有兩個指針分別指向兩個變量時,你可以通過某種機(jī)制交換這兩個指針的內(nèi)容,使得它們各自指向?qū)Ψ皆瓉淼奈恢?本介紹了關(guān)于C++地址交換的實現(xiàn),需要的朋友可以參考下
    2024-11-11
  • C++構(gòu)造函數(shù)初始化順序詳解

    C++構(gòu)造函數(shù)初始化順序詳解

    這篇文章主要介紹了C++構(gòu)造函數(shù)初始化順序詳解,是對C++代碼的運行機(jī)制深入探討,需要的朋友可以參考下
    2014-10-10
  • C/C++動態(tài)分配與釋放內(nèi)存的區(qū)別詳細(xì)解析

    C/C++動態(tài)分配與釋放內(nèi)存的區(qū)別詳細(xì)解析

    以下是對C與C++中動態(tài)分配與釋放內(nèi)存的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來參考下
    2013-09-09
  • C++中constexpr與模板元編程的基礎(chǔ)、常見問題、易錯點及其規(guī)避策略

    C++中constexpr與模板元編程的基礎(chǔ)、常見問題、易錯點及其規(guī)避策略

    C++編譯時計算允許程序在編譯階段完成計算任務(wù),constexpr與模板元編程是C編譯時計算的兩把利劍,它們不僅能夠提升程序的性能,還能增強(qiáng)代碼的健壯性和可維護(hù)性,通過避開本文闡述的易錯點,開發(fā)者可以更加得心應(yīng)手地運用這些特性,編寫出既高效又優(yōu)雅的C代碼
    2024-06-06
  • 解析C++中臨時對象的產(chǎn)生情況

    解析C++中臨時對象的產(chǎn)生情況

    臨時對象的產(chǎn)生和銷毀都是有成本的,都會影響程序的執(zhí)行性能和效率,所以如果能了解臨時對象產(chǎn)生的原因,就可以提升程序的性能和效率,下面小編就來和大家聊聊臨時對象產(chǎn)生的幾種情況吧
    2023-06-06
  • C語言 scanf輸入多個數(shù)字只能以逗號分隔的操作

    C語言 scanf輸入多個數(shù)字只能以逗號分隔的操作

    這篇文章主要介紹了C語言 scanf輸入多個數(shù)字只能以逗號分隔的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • c++實現(xiàn)高精度加法

    c++實現(xiàn)高精度加法

    高精度運算是指參與運算的數(shù)(加數(shù),減數(shù),因子……)范圍大大超出了標(biāo)準(zhǔn)數(shù)據(jù)類型(整型,實型)能表示的范圍的運算。例如,求兩個200位的數(shù)的和。這時,就要用到高精度算法了。
    2017-05-05
  • C語言實現(xiàn)學(xué)生信息管理程序

    C語言實現(xiàn)學(xué)生信息管理程序

    這篇文章主要為大家詳細(xì)介紹了C語言實現(xiàn)學(xué)生信息管理程序,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-03-03
  • 深入分析C++中聲明與定義的區(qū)別

    深入分析C++中聲明與定義的區(qū)別

    C++學(xué)了這么多年你知道為什么定義類時,類的定義放在.h文件中,而類的實現(xiàn)放在cpp文件中。它們?yōu)槭裁茨軌蜿P(guān)聯(lián)到一起呢?你知道什么東西可以放在.h文件中,什么不能。什么東西又可以放在cpp文件中。如果你忘記了或是壓根就不明白,那么讀過此文你會清晰無比?。?/div> 2014-09-09

最新評論