C語言函數(shù)棧幀的創(chuàng)建與銷毀詳解
前言
C語言,我們學(xué)到了這里,一定會(huì)有許多疑惑,我們的代碼邏輯,定義的變量,自定義的函數(shù)等通過編譯器,準(zhǔn)確的來講是集成開發(fā)環(huán)境(vs),預(yù)處理、編譯、匯編、鏈接,在電腦上執(zhí)行的過程中,計(jì)算機(jī)內(nèi)部到底是怎么運(yùn)行的,我們要搞清楚這些基礎(chǔ)的過程,而不是說,因?yàn)檫@樣做,所以有什么樣的結(jié)果,過程是怎么樣的,我們一概不了解,這樣學(xué)編程是不對(duì)的,要刨根問底,增強(qiáng)自己的內(nèi)功。保姆級(jí)講述,包您一看就會(huì),快來試試吧~
舉個(gè)例子:
int main() { int a=0; return 0; }
根據(jù)以上代碼,我們可以讀出一段信息,定義一個(gè)局部整型變量 a,并將他初始化為0,同時(shí)一個(gè)整型變量在內(nèi)存中占4個(gè)字節(jié)。關(guān)于main()函數(shù)的返回值啊,簡(jiǎn)單說一下,如果返回值為0,則說明程序是正常結(jié)束,非0,就是不正常結(jié)束,且返回值為整型數(shù)據(jù)。
那么博主在這里問大家兩個(gè)問題:
- 局部整型變量a,到底是怎么創(chuàng)建的?
- 如果我們對(duì)變量a不進(jìn)行初始化操作,局部變量a的值是多少?
- main()函數(shù)也有返回值,那他是不是也是也被其他函數(shù)調(diào)用?
函數(shù)在調(diào)用的過程中棧幀的創(chuàng)建和銷毀對(duì)于不同的編譯器來說略有差異,但大體的邏輯是差不多的。取決于編譯器。
什么是函數(shù)棧幀?
每一次函數(shù)的調(diào)用,操作系統(tǒng)都會(huì)在內(nèi)存的棧區(qū)上開辟一塊空間,稱為棧幀。
函數(shù)調(diào)用建立棧幀,棧幀中存儲(chǔ)局部變量,參數(shù)等等。
棧區(qū),堆區(qū)等是操作系統(tǒng)這門學(xué)科中對(duì)內(nèi)存的劃分,數(shù)據(jù)結(jié)構(gòu)的“棧”,“堆”是存放、處理數(shù)據(jù)的一種結(jié)構(gòu),跟內(nèi)存的棧區(qū),堆區(qū),沒有啥關(guān)系,但是有一點(diǎn),數(shù)據(jù)結(jié)構(gòu)的“棧”和內(nèi)存的棧區(qū)都是后進(jìn)先出,先進(jìn)后出的特性。
舉個(gè)例子:
在main() 函數(shù)內(nèi)部的Sum()函數(shù)則是被 main()函數(shù)調(diào)用,Sum()函數(shù)結(jié)束開辟的棧幀被操作系統(tǒng)回收,返回mian()函數(shù)的Sum()函數(shù)調(diào)用點(diǎn),繼續(xù)執(zhí)行程序。
程序順序執(zhí)行 ,執(zhí)行main()函數(shù),遇到 int a=1; 操作系統(tǒng)就會(huì)在內(nèi)存的棧區(qū)上,為main()函數(shù)創(chuàng)建棧幀里,給局部變量 a 分配一塊4個(gè)字節(jié)的空間。同理,局部變量在那個(gè)函數(shù)里創(chuàng)建的,那么就在那個(gè)函數(shù)的棧幀中為其分配空間。函數(shù)在調(diào)用的時(shí)候操作系統(tǒng)會(huì)為函數(shù)預(yù)劃分一片空間,就是所謂的棧幀,由 esp,ebp 兩個(gè)寄存器維護(hù),如果空間不夠了,操作系統(tǒng)會(huì)在這片空間的棧頂(esp)增加空間入棧,esp 可以動(dòng)態(tài)的維護(hù)棧頂。
知識(shí)點(diǎn)普及:
局部變量是在內(nèi)存的棧區(qū)上開辟的,棧區(qū)內(nèi)存的使用習(xí)慣是先使用高地址的空間,后使用低地址的空間,棧區(qū)又是一種先進(jìn)后出,后進(jìn)先出的結(jié)構(gòu),且只能在一端操作。
在為函數(shù)創(chuàng)建棧幀的時(shí)候,會(huì)有很多的寄存器去維護(hù),這一片的可使用空間就會(huì)被初始化為隨機(jī)值,這個(gè)根據(jù)不同的編譯器有不同的初始化方式,這就是我們對(duì)局部變量定義的時(shí)候不進(jìn)行初始化打印隨機(jī)值的原因。
由圖可見,棧底寄存器 ebp-0E4h 就等于 edi 指向的空間。
不過現(xiàn)在的編譯器很智能,如果沒有初始化局部變量使用的話會(huì)報(bào)錯(cuò),就很強(qiáng),但是大家還是要養(yǎng)成良好的初始化的習(xí)慣。
Sum() 函數(shù)的調(diào)用
傳參:
把變量a,b 的值分別存儲(chǔ)在 ecx ,eax 寄存器,分別入棧。就相當(dāng)于使用兩塊空間 又將a,b 存儲(chǔ)了一次。這就是形參嘛。
形參是實(shí)參的拷貝!這段形參開辟的空間實(shí)際上還是在main() 函數(shù)的棧幀中。esp 棧頂指針動(dòng)態(tài)維護(hù)。
esp ,edp 如何從main() 棧幀,來維護(hù) Sum() 函數(shù)棧幀:
區(qū)間實(shí)際上是(ebx,ebp)
return 會(huì)返回 call 指令的這個(gè)地址,可以繼續(xù)執(zhí)行程序。
C/C++ 程序內(nèi)存分配的幾個(gè)區(qū)域:
1.棧區(qū)(stack):在執(zhí)行函數(shù)時(shí),函數(shù)內(nèi)局部變量的存儲(chǔ)單元都可以在棧上創(chuàng)建,函數(shù)執(zhí)行結(jié)束時(shí)這些存儲(chǔ)單元自動(dòng)釋放。棧內(nèi)存分配運(yùn)算內(nèi)置于處理器的指令集中,效率很高,但是分配的內(nèi)存容量有限,棧區(qū)主要存放運(yùn)行函數(shù)而分配的局部變量,函數(shù)參數(shù),返回?cái)?shù)據(jù),返回地址等。
2.堆區(qū)(heap):一般由程序員分配釋放,若程序員不釋放,程序結(jié)束時(shí)由OS回收。
3.數(shù)據(jù)段(靜態(tài)區(qū))(static)存放全局變量,靜態(tài)變量。程序結(jié)束后由系統(tǒng)釋放。
4.代碼段:存放函數(shù)體(類成員函數(shù)和全局函數(shù))的二進(jìn)制代碼
至此C語言函數(shù)棧幀的創(chuàng)建和銷毀博主已經(jīng)分享完了,相信大家對(duì)這個(gè)函數(shù)怎么建立的、銷毀的,函數(shù)怎么傳參,使用等有了一定的理解,大家可以自己使用匯編代碼,感受一下。
總結(jié)
到此這篇關(guān)于C語言函數(shù)棧幀的創(chuàng)建與銷毀的文章就介紹到這了,更多相關(guān)C語言函數(shù)棧幀內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用C語言編寫一個(gè)強(qiáng)制關(guān)機(jī)程序
這篇文章主要為大家詳細(xì)介紹了如何使用C語言實(shí)現(xiàn)一個(gè)簡(jiǎn)單的"流氓軟件",一個(gè)可以強(qiáng)制關(guān)機(jī)惡作劇關(guān)機(jī)程序,輸入指定指令才可以解除,感興趣的小伙伴可以學(xué)習(xí)一下2023-11-11C++中對(duì)象的賦值與復(fù)制操作詳細(xì)解析
對(duì)象之間的賦值也是通過賦值運(yùn)算符“=”進(jìn)行的。本來賦值運(yùn)算符“=”只能用來對(duì)單個(gè)的變量賦值,現(xiàn)在被擴(kuò)展為兩個(gè)同類對(duì)象之間的賦值,這是通過對(duì)賦值運(yùn)算符的重載實(shí)現(xiàn)的2013-10-10C語言修煉之路數(shù)據(jù)類型悟正法?解析存儲(chǔ)定風(fēng)魔上篇
使用編程語言進(jìn)行編程時(shí),需要用到各種變量來存儲(chǔ)各種信息。變量保留的是它所存儲(chǔ)的值的內(nèi)存位置。這意味著,當(dāng)您創(chuàng)建一個(gè)變量時(shí),就會(huì)在內(nèi)存中保留一些空間。您可能需要存儲(chǔ)各種數(shù)據(jù)類型的信息,操作系統(tǒng)會(huì)根據(jù)變量的數(shù)據(jù)類型,來分配內(nèi)存和決定在保留內(nèi)存中存儲(chǔ)什么2022-02-02C++基于easyx圖形庫實(shí)現(xiàn)打磚塊游戲
這篇文章主要為大家詳細(xì)介紹了C++基于easyx圖形庫實(shí)現(xiàn)打磚塊游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-05-05C++中類的成員函數(shù)及內(nèi)聯(lián)函數(shù)使用及說明
這篇文章主要介紹了C++中類的成員函數(shù)及內(nèi)聯(lián)函數(shù)使用及說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11C++簡(jiǎn)明分析講解引用與函數(shù)提高及重載
今天繼續(xù)開始對(duì)C++核心編程知識(shí)的分享與系統(tǒng)講解,第一,這里會(huì)提到“引用”方法傳參以及剖析引用的本質(zhì);第二,我們對(duì)函數(shù)來一個(gè)提高,相當(dāng)于進(jìn)階函數(shù)了,包括函數(shù)的默認(rèn)值,簡(jiǎn)單的提一下函數(shù)的占位參數(shù),函數(shù)重載以及注意事項(xiàng),接下來上正文2022-05-05OpenCV實(shí)現(xiàn)拼接圖像的簡(jiǎn)單方法
這篇文章主要為大家詳細(xì)介紹了OpenCV實(shí)現(xiàn)拼接圖像的簡(jiǎn)單方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-05-05