成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

為什么C++中有函數(shù)指針還需要std::function?

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

大家好,我是小風(fēng)哥。

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

#include <stdio.h>

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

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

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

0000000000400526 <func>:
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這個(gè)地址,讓我們記住這個(gè)地址。

然后運(yùn)行一下編譯后生成的程序,想一想這段代碼會(huì)輸出什么呢?

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

$ ./a.out
0x400526

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

圖片

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

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

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

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

圖片

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

圖片

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

typedef void (*func) (int);

struct closure{
func f;
int arg;
};

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

  • 一個(gè)指向代碼的指針變量
  • 一個(gè)保存數(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ù)運(yùn)行依賴的數(shù)據(jù):

圖片

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

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

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

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

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

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

責(zé)任編輯:武曉燕 來源: 碼農(nóng)的荒島求生
相關(guān)推薦

2024-02-22 10:34:00

NULLC++nullptr

2024-02-18 12:39:15

C++autodecltype

2023-09-12 14:02:30

數(shù)組vector

2023-09-14 16:02:27

2023-10-24 15:15:26

HTTPWebSocket

2021-10-12 18:48:07

HTTP 協(xié)議Websocket網(wǎng)絡(luò)通信

2020-12-11 07:39:37

RPC MQ架構(gòu)

2020-04-26 09:32:13

物聯(lián)網(wǎng)安全技術(shù)

2021-03-31 13:20:02

智慧城市物聯(lián)網(wǎng)5G

2025-04-09 11:15:00

服務(wù)熔斷服務(wù)降分布式系統(tǒng)

2013-04-12 09:38:36

網(wǎng)絡(luò)設(shè)備IPv6網(wǎng)絡(luò)架構(gòu)

2015-10-19 11:06:37

2021-07-16 07:21:45

C++可調(diào)用對象std::functi

2023-06-01 07:50:42

JSDocTypeScriptAPI

2016-01-28 10:04:09

Jenkins運(yùn)維持續(xù)交付

2017-05-09 14:21:15

4G5G互聯(lián)網(wǎng)

2020-05-13 15:57:59

聚類分析算法監(jiān)督學(xué)習(xí)

2025-01-07 14:36:12

2020-12-15 06:57:24

java服務(wù)器

2017-12-11 17:14:56

ERP管理數(shù)字化
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 精品香蕉一区二区三区 | 欧美一级欧美一级在线播放 | 成人精品一区二区三区四区 | 国产精品久久久久久亚洲调教 | 亚洲三区在线观看 | 久久国产美女视频 | 久久久久久久久久久久久91 | 99re6在线| 国产亚洲精品精品国产亚洲综合 | 91视频正在播放 | 久色网| 久久久久久九九九九 | 美国av片在线观看 | 久久久久国产一区二区三区不卡 | 国产黄色在线观看 | 欧美日高清| 国产99久久 | 久久久精品综合 | 久久久久资源 | 中文字幕不卡视频在线观看 | 亚洲国产一区二区三区四区 | 国产精品免费在线 | 在线观看亚洲专区 | 亚洲性视频网站 | 青青草av网站 | 精久久久 | 久久精品91久久久久久再现 | 亚洲成人中文字幕 | 日韩中文一区二区三区 | 欧美精品日韩精品国产精品 | 成人在线播放网站 | 久久久久亚洲精品 | 久久91精品久久久久久9鸭 | 久草在线| 日本精品在线播放 | 一级a爱片性色毛片免费 | 日本久草| 91久久久久久久久久久 | 国产丝袜人妖cd露出 | 国产精品大片在线观看 | 最新中文字幕一区 |