日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
為什么C++中有函數(shù)指針還需要std::function?

大家好,我是小風哥。

我們提供的服務有:網(wǎng)站設計制作、成都網(wǎng)站建設、微信公眾號開發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認證、平陰ssl等。為千余家企事業(yè)單位解決了網(wǎng)站和推廣的問題。提供周到的售前咨詢和貼心的售后服務,是有科學管理、有技術(shù)的平陰網(wǎng)站制作公司

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

#include 

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

void main() {
int (*f)(int) = func;
printf("%p\n", 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這個地址,讓我們記住這個地址。

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

顯然應該是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)在你應該明白函數(shù)指針了,細心的同學可能會有一個疑問,為什么編譯器在生成可執(zhí)行文件時就知道函數(shù)func存放在內(nèi)存地址0x400526上呢?這不應該是程序被加載到內(nèi)存后開始運行時才能確定的嗎?

函數(shù)指針的作用是可以把一段代碼當做一個變量傳來傳去,主要的用途之一就是回調(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ù)簽名即可。


新聞標題:為什么C++中有函數(shù)指針還需要std::function?
本文來源:http://m.5511xx.com/article/ccopces.html