36.【C语言】函数栈帧的创建和销毁

本文将解决以下问题

局部变量是怎么创建的?
为什么局部变量的值是随机值?
函数是怎么传参的?传参的顺序是怎样的?
形参和实参是什么关系?
函数调用是怎么做的?
函数调用是结束后怎么返回的?

本文使用VS2013查看 ,从汇编语言来理解

1.基本概念

寄存器作用:寄存数据

eax:累加器 ebx:作为指针指向数据 ecx:循环计数器 edx:eax的扩展

地址寄存器:ebp,esp(维护函数栈帧)

注意:每一个函数的调用,都会在栈区创造一个空间

画图解释

#include <stdio.h>
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);
	getchar();
	return 0;
}

2.main函数的调用

按F11后点击调用堆栈

执行完后:

//crtexe.c的代码
#endif  /* _WINMAIN_ */
        void
        )
{
        /*
         * The /GS security cookie must be initialized before any exception
         * handling targetting the current image is registered.  No function
         * using exception handling can be called in the current image until
         * after __security_init_cookie has been called.
         */
        __security_init_cookie();

        return __tmainCRTStartup();
}
……………………
static
int
__tmainCRTStartup(
         void
         );
……………………
#else  /* WPRFLAG */
            __initenv = envp;
            mainret = main(argc, argv, envp);//main函数被调用

则调用关系为mainCRTStartup-->__tmainCRTStartup-->main

3.从反汇编上理解

按F11后右击反汇编,选择不显示符号名

#include <stdio.h>
int add(int x, int y)
{
000213C0  push        ebp  
000213C1  mov         ebp,esp  
000213C3  sub         esp,0CCh  
000213C9  push        ebx  
000213CA  push        esi  
000213CB  push        edi  
000213CC  lea         edi,[ebp+FFFFFF34h]  
000213D2  mov         ecx,33h  
000213D7  mov         eax,0CCCCCCCCh  
000213DC  rep stos    dword ptr es:[edi]  
	int z = 0;
000213DE  mov         dword ptr [ebp-8],0  
	z = x + y;
000213E5  mov         eax,dword ptr [ebp+8]  
000213E8  add         eax,dword ptr [ebp+0Ch]  
000213EB  mov         dword ptr [ebp-8],eax  
	return z;
000213EE  mov         eax,dword ptr [ebp-8]  
}
000213F1  pop         edi  
000213F2  pop         esi  
000213F3  pop         ebx  
000213F4  mov         esp,ebp  
000213F6  pop         ebp  
000213F7  ret 
--------------------------------------------------------------------
int main()
{
00021410  push        ebp  
00021411  mov         ebp,esp  
00021413  sub         esp,0E4h  
00021419  push        ebx  
0002141A  push        esi  
0002141B  push        edi  
0002141C  lea         edi,[ebp+FFFFFF1Ch]  
00021422  mov         ecx,39h  
00021427  mov         eax,0CCCCCCCCh  
0002142C  rep stos    dword ptr es:[edi]  
	int a = 10;
0002142E  mov         dword ptr [ebp-8],0Ah  
	int b = 20;
00021435  mov         dword ptr [ebp-14h],14h  
	int c = 0;
0002143C  mov         dword ptr [ebp-20h],0  
	c = add(a, b);
00021443  mov         eax,dword ptr [ebp-14h]  
00021446  push        eax  
00021447  mov         ecx,dword ptr [ebp-8]  
0002144A  push        ecx  
0002144B  call        000210E6  
00021450  add         esp,8  
00021453  mov         dword ptr [ebp-20h],eax  
	printf("%d\n",c);
00021456  mov         esi,esp  
00021458  mov         eax,dword ptr [ebp-20h]  
0002145B  push        eax  
0002145C  push        25858h  
00021461  call        dword ptr ds:[00029114h]  
00021467  add         esp,8  
0002146A  cmp         esi,esp  
0002146C  call        0002113B  
	return 0;
00021471  xor         eax,eax  
}
00021473  pop         edi  
00021474  pop         esi  
00021475  pop         ebx  
00021476  add         esp,0E4h  
0002147C  cmp         ebp,esp  
0002147E  call        0002113B  
00021483  mov         esp,ebp  
00021485  pop         ebp  
00021486  ret  

图片详解:

4.视频演示

36.【C语言】函数栈帧的创建和销毁文章的汇编演示

4.往期推荐

35.【C语言】详解函数递归

28.【C语言】函数系列上 库函数

29.【C语言】函数系列中 自定义函数

30.【C语言】函数系列下

  • 22
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值