修炼内功——理解函数栈帧创建和销毁
函數(shù)棧幀
1. 全局觀:
今天我們講的是棧區(qū),以下的圖上方是低地址,下方是高地址,跟上圖相反,因此使用順序是從下方到上方,跟上圖本質(zhì)上一致。
2. 函數(shù)的調(diào)用關(guān)系
ebp,esp這兩個(gè)寄存器中存放的是地址,這兩個(gè)地址是用來(lái)維護(hù)函數(shù)棧幀的。調(diào)用哪個(gè)函數(shù)就維護(hù)哪個(gè)函數(shù)棧幀
每一個(gè)函數(shù)調(diào)用都要在棧區(qū)開(kāi)辟空間。
在VS2013中,main函數(shù)也是被其他函數(shù)調(diào)用的,調(diào)用關(guān)系如下圖
3. 分析一段代碼
int Add(int x, int y) {int z = 0;z = x + y;return z; }int main() {int a = 10;int b = 20;int c = 0;c = Add(a, b);printf("%d\n",c);return 0; }分析main函數(shù)的反匯編
到此為止,main函數(shù)開(kāi)辟好了
Add函數(shù)調(diào)用前的準(zhǔn)備
因此我們講形參是實(shí)參的一份臨時(shí)拷貝
存放call指令下一條指令的地址
調(diào)用Add函數(shù)
Add返回值
每pop一次esp往下方走一次
然后ebp和esp都回到了main函數(shù)開(kāi)辟后的地方,再通過(guò)存著的地址找到了call的下一條語(yǔ)句再繼續(xù)執(zhí)行
然后把return的值賦值給c
打印出來(lái)就結(jié)束啦
4. 總結(jié)
這里我們可以了解到除了蹦出去還要留下地址供自己找回來(lái)
以及為什么不初始化的變量會(huì)打印出“燙燙燙”(因?yàn)殚_(kāi)辟時(shí)會(huì)把空間初始化成cccccccc)
還有傳值調(diào)用時(shí)形參是實(shí)參的一份拷貝,因此會(huì)比傳指調(diào)用占用更多空間
希望大家能更深層理解函數(shù)棧幀創(chuàng)建和銷毀
總結(jié)
以上是生活随笔為你收集整理的修炼内功——理解函数栈帧创建和销毁的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 地图裁剪器,可以将图片裁剪成瓦片数据,主
- 下一篇: 10个免费的顶级跨浏览器测试工具