从汇编角度看函数调用过程
创始人
2025-06-01 17:53:25
0

本文以一个简单的程序为例,通过汇编代码查看函数调用过程,涉及如何开辟栈帧,函数如何返回等

#include 
using namespace std;int sum(int a, int b)
{int temp = 0;temp = a + b;return temp;
}int main()
{int a = 10;int b = 20;int ret = sum(a, b);cout << "ret: " << ret << endl;return 0;
}

代码非常简单,调用一个sum函数计算两数之和。

下面通过vs2017调试代码,看看代码编译成汇编指令后,是如何开辟栈帧及函数调用的。

(通过vs调试时记得把编译优化选项关了)

这是main函数部分汇编代码:

int main()
{
00271630  push        ebp  
00271631  mov         ebp,esp  
00271633  sub         esp,0Ch  int a = 10;
00271636  mov         dword ptr [ebp-8],0Ah  int b = 20;
0027163D  mov         dword ptr [ebp-4],14h  int ret = sum(a, b);
00271644  mov         eax,dword ptr [ebp-4]  
00271647  push        eax  
00271648  mov         ecx,dword ptr [ebp-8]  
0027164B  push        ecx  
0027164C  call        00271610  
00271651  add         esp,8  
00271654  mov         dword ptr [ebp-0Ch],eax cout << "ret: " << ret << endl;//.....
}

(vs可能会用变量名代替地址,比如mov dword ptr [a],0Ah,只需要右键取消勾选显示符号名即可看到源地址)

可以看到进入main函数做的第一件事就是开辟栈帧,push ebp将栈底指针压栈,mov ebp esp将·esp复制给ebp,即将栈底指针指向栈顶,sub esp,0Ch栈顶指针减去0Ch,实则就是开辟栈空间。(栈向低地址增长)
在这里插入图片描述
随后将0Ah赋值给dword ptr[ebp-8]0Ah是10,也就是a变量,b同理。
在这里插入图片描述
接着调用sum函数,在调用函数之前这里进行了参数压栈操作,先将b的值给到eax,再进行压栈,a同理。
在这里插入图片描述
看到这里我们可以得出一个结论,参数是在函数调用方压栈的。

下面的call指令,调用sum函数,值得注意的是,调用call函数之前会将call后面的指令地址(00271651)压栈,为了在函数返回时继续向下执行。
在这里插入图片描述
进入到sum函数,这是sum函数汇编代码:

int sum(int a, int b)
{
00271610  push        ebp  
00271611  mov         ebp,esp  
00271613  push        ecx  int temp = 0;
00271614  mov         dword ptr [ebp-4],0  temp = a + b;
0027161B  mov         eax,dword ptr [ebp+8]  
0027161E  add         eax,dword ptr [ebp+0Ch]  
00271621  mov         dword ptr [ebp-4],eax  return temp;
00271624  mov         eax,dword ptr [ebp-4]  
}
00271627  mov         esp,ebp  
00271629  pop         ebp  
0027162A  ret  

在sum函数中,先将ebp压栈,再让ebp指向esp的位置。后将ecx压栈:
在这里插入图片描述
再往下mov dword ptr [ebp-4],0 ecx的位置赋为0。mov eax,dword ptr [ebp+8]ebp + 8刚好指向a的位置,将a的值移到eax寄存器中。add eax,dword ptr [ebp+0Ch]连加上b的值,存放在eax中。mov dword ptr [ebp-4],eaxeax的值移到ebp - 4的位置,最后在函数返回时,mov eax,dword ptr [ebp-4]将结果存放在eax寄存器中。之后进行函数返回的操作,mov esp,ebp
在这里插入图片描述
pop ebp,弹出值给到ebp,而现在栈顶的值刚好是之前保存的ebp的值:
在这里插入图片描述
ret指令首先弹出栈顶元素,并把弹出的内容放到PC寄存器中:
在这里插入图片描述
PC寄存器中存放的是下一条要执行的指令的地址,一个神奇的事情是,刚刚弹出的地址(00271651)刚好是call指令的下一条指令,也就是执行完sum函数后的下一条指令。这也就解释了函数调用完是怎么接着往后执行的。

回到main函数中,add esp,8将栈顶指针加8,回退栈顶指针,“回收”临时的函数参数。
在这里插入图片描述
这时回到最初的起点,mov dword ptr [ebp-0Ch],eax将计算所得的结果给到dword ptr [ebp-0Ch]。至此函数执行完成。可见,其实并不复杂,当然示例比较简单,但道理都一样。清楚了整个函数调用过程,或许就能更好理解为什么不要返回局部变量的地址?

int *fun()
{int temp = 5;return &temp;
}int main()
{int *p;p = fun();//为什么不要这样做?cout << *p << endl;return 0;
}

因为在函数执行完成后,栈帧已经交还给了系统,虽然这时可以得到正确结果,但这只是因为系统没有对栈帧内容清空。

如果在打印*p之前调用一下函数sum(1, 2),这时结果就是不确定的了。

int main()
{int *p;p = fun();sum(1, 2);cout << *p << endl;return 0;
}
//输出8322272

所以不要返回局部变量地址,即便当时程序没有报错!!!

清楚了整个函数调用过程,或许就能更好理解为什么有时未初始化的数据在调试模式下显示“烫”或者“屯”,这是因为,开辟的栈空间的每一个字节默认初始化为0xCC,而0xCCCC的汉字编码就是“烫”。有时编译器还会使用0xCDCD来初始化,这时看到的就是“屯”。

清楚了整个函数调用过程,或许就能更好理解栈非法访问以及爆栈的问题(一个进程的栈空间默认为8M左右,可以修改大小,记得之前面试被问过这个问题)。

看看ChatGPT给出的解释:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

相关内容

热门资讯

强外贸拓市场 宜宾19家企业亮... 4月15日至5月5日,第139届中国进出口商品交易会(简称“广交会”)在广州举办。活动期间,宜宾市商...
原创 运... 运-20运输机近日再一次展现了其非凡能力——为救治一名身处西藏阿里高原的重病边防战士,它完成了一次跨...
水井坊总经理再变动!胡庭洲在任... 在刚刚披露完2025年报和2026年一季报后,水井坊再次迎来核心高管的变动。 5月6日,水井坊发布公...
康宁公司(GLW)维持超过14... 康宁公司(GLW)维持超过14%的涨幅,持稳于185美元附近,美股盘初曾达到195.81美元——盘中...
原创 国... 阅读须知:本文内容所有信息和数据,均为作者查阅官方信息和网络已知数据整合解析,旨在让读者更清晰了解相...
原创 金... 2026年5月6日,国内金价继续从高位回落,上海黄金交易所的AU9999报每克1013元,沪金期货主...
新华鲜报丨交易笔数大增 从支付...   新华社北京5月6日电(记者吴雨)消费市场活力十足,尽显中国经济强劲韧性。中国人民银行5月6日发布...
原创 安... - 对外直接投资[1]:2026年一季度中国全行业对外直接投资445亿美元,同比增长8.9%;非金融...
龙芯中科:现在公司主打性价比,... 证券日报网5月6日讯 ,龙芯中科在接受调研者提问时表示,公司过去主打芯片自主性,所以是应用导向的,比...
新华鲜报|交易笔数大增 从支付... 消费市场活力十足,尽显中国经济强劲韧性。中国人民银行5月6日发布的数据显示,今年“五一”假期支付交易...
马斯克同意支付150万美元罚款 据路透社报道,4日提交的一份法庭文件显示,美国证券交易委员会已经与美国企业家马斯克就收购推特期间涉嫌...
福建沙县农商银行被罚170万元 【大河财立方消息】5月6日,国家金融监督管理总局三明监管分局披露的行政处罚信息显示,因违规下达存款考...
原创 只... 此前盛传三星要暴砍中国市场产品线的消息,终于落地了。 5月6日,三星在官网正式发布公告,停止在中国大...
珀莱雅上市9年首次“双降”,二... 出品 |达摩财经 此次是珀莱雅第二次向港交所递表。去年8月,珀莱雅发布公告称,为加快国际化战略和海...
沿河县中界镇创新“三营联动”发... 沿河县中界镇创新“三营联动” 发展壮大农村集体经济 近年来,沿河县中界镇聚焦“强村富民”目标,创新推...
老牌私募打新违规被“点名”! 来源:金融时报 一家老牌私募机构因网下打新违规被“点名”。 近日,中国证券业协会(以下简称“中证协”...
和讯信息张杨:五月开门红!存储... 5月开门红,存储芯片大涨还能不能在科技里躺平了?和讯信息张杨分析,首先是假期外围大涨,导致今天芯片大...
警报响起!30年期美债收益率再... 全球股市处于历史高位,但债券市场的警示信号越来越清晰。 30年期美国国债收益率重新站上5%的关口,再...
五粮液急了!先推80亿元至10... 年报推迟公布之后,五粮液摊上事了。股民纷纷质疑五粮液“修改数据”,二级市场,公司股价今日逆市下跌近5...
收假必看!央行+海关新政落地,... 来源:市场资讯 (来源:矿业俱乐部) 央行、海关总署联合发布公告,自2026年6月1日起升级黄金及...