網站首頁 編程語言 正文
正片開始
今天來講講我對棧幀創建與銷毀的拙見。
理解什么是棧幀首先知道什么是棧:
在數據結構中, 棧是限定僅在表尾進行插入或刪除操作的線性表。棧是一種數據結構,它按照后進先出的原則存儲數據,先進入的數據被壓入棧底,最后的數據在棧頂,需要讀數據的時候從棧頂開始彈出數據。
棧有什么用?
在計算機系統中,棧也可以稱之為棧內存是一個具有動態內存區域,存儲函數內部(包括main函數)的局部變量和方法調用和函數參數值,是由系統自動分配的,一般速度較快;存儲地址是連續且存在有限棧容量,會出現溢出現象程序可以將數據壓入棧中,也可以將數據從棧頂彈出。壓棧操作使得棧增大,而彈出操作使棧減小。
棧用于維護函數調用的上下文,離開了棧函數調用就沒法實現。
講到這里,小朋友你是否有很多問號?那打住,我們拋開無聊的學術前文,另起爐灶。
寄存器
要講清楚棧幀就必須理解一手寄存器。尤其是 ebp,esp這2個寄存器中存放的地址,這兩個地址是用來維護函數棧幀的。
寄存器有很多種這里不贅述
main函數創建
我們這里隨便搞一個最簡單的Add函數
int add(int x,int y) { int z; z=x+y; return z; } int main() { int data1; int data2; int ret; while(1) { int data1,data2 = 0; scanf("%d %d",&data1,&data2); add(data1,data2); return 0; }
搞棧幀的話我的編譯器是不適合的,我是vs2019,因為編譯器越高級函數的封裝越復雜周密,不容易我們去剖析棧幀,我就盡量語言表達嚴謹一點吧。編譯器反匯編過程就能反應我們棧幀創建的過程,這是我在網上找的反匯編頁面可以參考一下
其中反匯編用到的指針我們要清楚意義:
在編譯器中,main函數也是會被其他函數調用的,調用堆棧窗口后反匯編可以看到如下字樣:
main _tmainCRTStartup mainCRTStartup
后面兩句意義不明的玩意兒就是在調用main函數。為什么要講這個呢?我們說每一次函數調用都要分配空間,main函數不例外也要分配棧幀空間。
以下內容和上面匯編指令表食用更佳:
首先 push ,即壓棧,就是往棧sei東西進去。push 會讓esp讓低地址走,就會在原先基礎上壓進來一個 ebp 指針。
接下是 mov 指針,mov把后面的指針賦到前面去,esp給了ebp,也就是相當于在移位。
接下來是 sub 減法操作,減去一個內容來使esp指針走向低地址來開辟main函數棧幀。
過程模擬如下:
局部變量創建
接下來esp已經走到那幾個內容的頭上去了,這時出現了 lea 指針,即 load effective address 加載有效地址,其實在這個指針指定對象里面放入一個地址
我們后面的 [ebp-0C0h],其實就是剛剛 sub操作,本質上還是原來開辟棧幀起點 ebp 的地址,把這個地址放入edi 里面。
接下來的連續 mov 時在把從edi 開始的 30h 這么多個空間里面的 dword(double word-四字節數據)全部初始化成 eax 里面 “0CCCCCCCCh”的內容,保證為main函數預開辟的內存全變成 “CCCCCCCCh”,這么說來改的還是蠻多的。
接下來當我們創建變量時,比如 int a = 10;就會出現類似下面字樣:
int a = 10; 00C2142E C7 45 EC 0A 00 00 00 mov dword ptr [ebp-8h],0Ah
這里就是在創建局部變量了, ebp指針減了 8h,這個 8h 就是給a留的位子**(這里的 h 是編譯器給的標識,我們只需要明白這是一個十六進制數)**就行了。所以總結一下,其實創建方式與main函數沒有太大出入。
函數部分
Add函數傳參時也是在將 esp 進行壓棧,但注意,這時的esp里面的值是 10,相當于是在傳 10 這個值。傳完參緊接著就會調用函數
00C2144B E8 91 FC FF FF call 00C210E1
call 指針作用就是調用函數,F11 執行call指令后會發現在跳轉到作用的同時,他會把 call指令的下一條指令的地址傳到里面,在頂上壓一個main函數的ebp ,esp又會跑到最上面,一但函數執行完后返回就會很自然的回到該地址。
在main函數的 ebp 上面又會傳統藝能,以相同的方式開辟 Add 函數的空間,又初始化成全 c,以相同方式創建臨時變量……
這時你可能會注意到傳進函數的 x,y去哪里了?其實已經為他準備好了,在返回進行下一項指令時,x,y就會乖乖跑到這片空間儲存
Add函數完成后回把傳的參返回, 就是我們的 pop 指針,即出棧,這里參數每從棧頂pop一次 esp 指針就會上移一個單位,ebp也會隨之退回一個單位,利用指針的偏移量找回他的形參,最后返回值ret,其邏輯本質上就是彈出main ebq那里的下一項指令的地址。
我們走出函數后,esp,ebq會回收,這時這塊空間就會直接銷毀,挫骨揚灰。
整個函數部分就完美的呈現出來了。
形參與實參
形參確實是我在壓棧時開辟的空間,這坨空間是獨立的,只是值是相同的,形參是實參的一份臨時拷貝,改變形參不影響實參,那返回值是怎么帶回來的呢?其實是通過寄存器。
原文鏈接:https://blog.csdn.net/qq_61500888/article/details/121417241
相關推薦
- 2023-02-09 C++存儲鏈接性原理詳解_C 語言
- 2022-11-05 Nginx反向代理location和proxy_pass配置規則詳細總結_nginx
- 2022-04-17 C# 利用ExcelDataReader 讀取excel文件
- 2022-07-30 Python?excel合并居中值相同的單元格實例代碼_python
- 2024-01-31 成功解決Unable to negotiate with 182.92.119.208 port 2
- 2022-07-26 Android自定義評分控件的完整實例_Android
- 2022-11-23 Python字典高級用法深入分析講解_python
- 2022-07-22 使用@ControllerAdvice和@ExceptionHandler構建全局異常處理器
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支