0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

C++中有函數(shù)指針還需要std::function嘛

jf_78858299 ? 來源:碼農(nóng)的荒島求生 ? 作者:陸小風(fēng) ? 2023-02-15 14:13 ? 次閱讀

C/C++中可以使用指針指向一段代碼,這個指針就叫函數(shù)指針,假設(shè)有這樣一段代碼:

#include 

int func(int a) {
  return a + 1;
}

void main() {
   int (*f)(int) = func;
   printf("%p
", f);
}

我們定義了一個函數(shù)func,然后使用指針變量f指向該函數(shù),然后打印出變量f指向的地址,代碼很簡單,然后我們編譯一下,看下編譯后生成的指令,我們重點關(guān)注func函數(shù):

0000000000400526 :
  400526:       55                      push   %rbp
  400527:       48 89 e5                mov    %rsp,%rbp
  40052a:       89 7d fc                mov    %edi,-0x4(%rbp)
  40052d:       8b 45 fc                mov    -0x4(%rbp),%eax
  400530:       83 c0 01                add    $0x1,%eax
  400533:       5d                      pop    %rbp
  400534:       c3                      retq

可以看到,編譯好后的函數(shù)func位于地址0x400526這個地址,讓我們記住這個地址。

然后運行一下編譯后生成的程序,想一想這段代碼會輸出什么呢?

顯然應(yīng)該是func函數(shù)的在內(nèi)存中的地址!

$ ./a.out
0x400526

沒有猜錯吧,實際上函數(shù)指針本質(zhì)也是一個指針,只不過這個指針指向的不是內(nèi)存中的一段數(shù)據(jù)而是內(nèi)存中的一段代碼,就像這樣:

圖片

看到了吧,我們常說的指針一般都是指向內(nèi)存中的一段數(shù)據(jù),而函數(shù)指針指向了內(nèi)存中的一段代碼,在這個示例中指向了內(nèi)存地址0x400526,在這個地址中保存了函數(shù)func的機器指令。

現(xiàn)在你應(yīng)該明白函數(shù)指針了,細(xì)心的同學(xué)可能會有一個疑問,為什么編譯器在生成可執(zhí)行文件時就知道函數(shù)func存放在內(nèi)存地址0x400526上呢?這不應(yīng)該是程序被加載到內(nèi)存后開始運行時才能確定的嗎?

這個問題的答案收錄在了星球中,星球里的同學(xué)查看最新一期的文章即可。

圖片

函數(shù)指針的作用是可以把一段代碼當(dāng)做一個變量傳來傳去,主要的用途之一就是回調(diào)函數(shù),關(guān)于回調(diào)函數(shù)你可以參考《回調(diào)函數(shù)的實現(xiàn)原理》這篇文章。

關(guān)于回調(diào)函數(shù)其實是在A模塊定義,在B模塊被調(diào)用,就像這樣:

圖片

然而有時我們會有這樣的場景,我們依然需要在模塊A定義函數(shù),同時函數(shù)A的運行需要依賴B模塊產(chǎn)生的數(shù)據(jù),然后將模塊A定義的函數(shù)和模塊B產(chǎn)生的數(shù)據(jù)一并傳遞給C模塊來調(diào)用,就像這樣:

圖片

此時,單純的函數(shù)指針已經(jīng)不夠用了,因為函數(shù)指針只是單純的指向了內(nèi)存中的一段代碼,我們不但需要將內(nèi)存中的一段代碼同時也需要將內(nèi)存中的一塊數(shù)據(jù)傳遞給模塊C, 此時你可以定義一個結(jié)構(gòu)體,將代碼和數(shù)據(jù)打包起來 ,就像這樣:

typedef void (*func) (int);

struct closure{
  func f;
  int arg;    
};

我們將這個結(jié)構(gòu)體命名為closure,注意看,這個結(jié)構(gòu)中有兩部分:

  • 一個指向代碼的指針變量
  • 一個保存數(shù)據(jù)的變量

這樣,我們在A模塊為指針變量賦值,在B模塊為保存數(shù)據(jù)的變量賦值,然后將此結(jié)構(gòu)體傳遞給模塊C,模塊C中可以這樣使用:

void run(struct functor func) {
    func->f(func->arg);
}

即,closure既包含了一段代碼也包含了這段代碼使用的數(shù)據(jù),這里的數(shù)據(jù)也被稱為context,即上下文,或者environment,即環(huán)境,不管怎么稱呼,其實就是函數(shù)運行依賴的數(shù)據(jù):

圖片

而這也正是C++中std::function的目的所在。

單純的函數(shù)指針并沒有捕捉上下文的能力 ,這里的上下文就是指代碼依賴的數(shù)據(jù),你不得不自己動手構(gòu)造出一個結(jié)構(gòu)體用來存儲代碼依賴的上下文。

在C++中你沒有辦法單純的利用函數(shù)指針指向?qū)ο蟮某蓡T函數(shù),就是因為函數(shù)指針沒有辦法捕捉this(指向?qū)ο蟮闹羔?這個上下文。

std::function的作用本質(zhì)上和我們剛才定義的結(jié)構(gòu)體區(qū)別不大。

利用std::function你不但可以保存一段代碼,同時也可以保存必要的上下文,然后在合適的地方基于上下文調(diào)用這段代碼。

同時std::function也更加通用,你可以用其存儲任何可以被調(diào)用的對象(callable object),只要有正確的函數(shù)簽名即可。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • C++
    C++
    +關(guān)注

    關(guān)注

    21

    文章

    2090

    瀏覽量

    73405
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4696

    瀏覽量

    68083
  • 函數(shù)指針
    +關(guān)注

    關(guān)注

    2

    文章

    56

    瀏覽量

    3766
收藏 人收藏

    評論

    相關(guān)推薦

    請問F28377D中CPU2還需要配置時鐘和FLASH嗎

    在F28377D的例程中,有些CPU2的主函數(shù)沒有包含InitSysCtrl();比如例程GPIO_toggle_cpu02.c但是有些例程中,CPU2中包含了InitSysCtrl();比如例程blinky_cpu02.c所以
    發(fā)表于 09-05 11:19

    C/C++回調(diào)函數(shù)

    列表);最后需要注意的是,指向函數(shù)指針變量沒有 ++ 和 -- 運算。C++類的靜態(tài)函數(shù)作為回調(diào)函數(shù)
    發(fā)表于 02-11 15:25

    嵌入式-C++函數(shù)的重載

    函數(shù)名定義,從而形成重載,方便使用,某種層度上函數(shù)的重載也是C++三大特性之一的多態(tài)的一種形式 三、函數(shù)重載的應(yīng)用場景 #include //利用using指令打開
    發(fā)表于 06-28 13:54

    C語言指針函數(shù)函數(shù)指針詳細(xì)介紹

    C語言指針函數(shù)函數(shù)指針詳細(xì)介紹。。。。。。。
    發(fā)表于 03-04 15:27 ?5次下載

    C++實驗--指針

    C++實驗--指針
    發(fā)表于 12-30 14:50 ?1次下載

    C++指針”學(xué)習(xí)建議

    C++指針”學(xué)習(xí)建議
    發(fā)表于 03-31 15:53 ?3次下載

    c語言函數(shù)指針定義,指針函數(shù)函數(shù)指針的區(qū)別

     往往,我們一提到指針函數(shù)函數(shù)指針的時候,就有很多人弄不懂。下面就由小編詳細(xì)為大家介紹C語言中函數(shù)
    發(fā)表于 11-16 15:18 ?3596次閱讀

    C++程序設(shè)計教程之函數(shù)機制的詳細(xì)資料說明

    本文檔詳細(xì)介紹的是C++程序設(shè)計教程之函數(shù)機制的詳細(xì)資料說明主要內(nèi)容包括了: 1.函數(shù)性質(zhì)( Function Character ) , 2.指針
    發(fā)表于 02-22 11:24 ?2次下載
    <b class='flag-5'>C++</b>程序設(shè)計教程之<b class='flag-5'>函數(shù)</b>機制的詳細(xì)資料說明

    C++封裝:this指針

    C++封裝:this指針
    的頭像 發(fā)表于 06-29 14:37 ?3421次閱讀
    <b class='flag-5'>C++</b>封裝:this<b class='flag-5'>指針</b>

    C++ function技術(shù)的實現(xiàn)與具體運用

    當(dāng)std::function對象沒有初始化任何實際的可調(diào)用元素,調(diào)用std::function對象將拋出std::bad_
    的頭像 發(fā)表于 01-20 09:23 ?3289次閱讀

    std::function簡介及模板類聲明

    包裝下列這幾種可調(diào)用實體:函數(shù)、函數(shù)指針、成員函數(shù)、靜態(tài)函數(shù)、lamda表達(dá)式和函數(shù)對象。
    的頭像 發(fā)表于 07-28 15:30 ?4719次閱讀

    C++中如何用虛函數(shù)實現(xiàn)多態(tài)

    01 — C++函數(shù)探索 C++是一門面向?qū)ο笳Z言,在C++里運行時多態(tài)是由虛函數(shù)和純虛函數(shù)
    的頭像 發(fā)表于 09-29 14:18 ?1642次閱讀

    深度解析C++中的虛函數(shù)

    的重要手段。沒有虛函數(shù),C++C的區(qū)別就不大,都需要借助大量的“函數(shù)指針”,進(jìn)行面向?qū)ο蟮某绦?/div>
    的頭像 發(fā)表于 02-15 11:14 ?778次閱讀
    深度解析<b class='flag-5'>C++</b>中的虛<b class='flag-5'>函數(shù)</b>

    function與invoke的區(qū)別

    std::functionstd::invoke是兩個不同的東西,功能也不同。std::function 是一個
    的頭像 發(fā)表于 04-27 15:13 ?664次閱讀

    C++ std::tie函數(shù)的作用和用法

    C++std::tie函數(shù)的作用就是從元素引用中生成一個tuple元組,其在頭文件中定義
    的頭像 發(fā)表于 07-18 17:28 ?762次閱讀