日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁 > 技術(shù)學(xué)院 > 碼農(nóng)的荒島求生
[導(dǎo)讀]實(shí)際上函數(shù)指針本質(zhì)也是一個(gè)指針,只不過這個(gè)指針指向的不是內(nèi)存中的一段數(shù)據(jù)而是內(nèi)存中的一段代碼

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

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

#include  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:
  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í)才能確定的嗎?

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

函數(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)用的對(duì)象(callable object),只要有正確的函數(shù)簽名即可。

最后,我準(zhǔn)備開通知識(shí)星球啦,鼓勵(lì)大家在這里輸出自己的深度、系統(tǒng)性的思考,沉淀出知識(shí),我個(gè)人的力量畢竟有限,不可能把計(jì)算機(jī)中所有的主題都能系統(tǒng)性總結(jié)出來,但有了你們就不一樣了。
加入星球,你既可以選出成為和小風(fēng)哥一樣的輸出者(寫作),也可以只作為輸入者(閱讀),并努力進(jìn)化成為我們希望的樣子。
本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢抑制與過流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問題卻十分常見,不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車場照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉