深入理解C++函数栈帧

 更新时间:2021年07月09日 10:02:13   作者:物随心转  
本文主要介绍了C++函数栈帧,详细的介绍了C++函数栈帧的概念以及使用,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

一、什么是函数栈帧

每一次函数调用都是一个过程,为函数开辟栈空间,用于本次函数调用中临时变量的保存、现场保护。这块栈空间我们称为函数栈帧。栈是从高地址向低地址延伸的。寄存器ebp指向当前的栈帧的底部(高地址),寄存器esp指向当前的栈帧的顶部(低地址)

注:esp:栈指针寄存器(extended stack pointer),其内存放着一个指针,该指针永远指向系统栈最上面一个栈帧的栈顶。
bsp:基址指针寄存器(extended base pointer),其内存放着一个指针,该指针永远指向系统栈最上面一个栈帧的底部。

入栈push和出栈pop

push ebp就等于将ebp的值保存到栈中,并且将当前esp下移

pop ebp就等于将ebp的值从栈中取出来,将ebp指向这个值

内存空间大致可以用下图表示:

二、具体原理

我们来通过运行程序来分析,有如下程序

 int sum(int _a, int _b)
 {
	 int res = 0;
	 res = _a + _b;
 
	 return res;
 }
 
 
 int main()
 {
	 int a = 10;
	 int b = 20;
 
	 int ret = sum(a, b);
	 printf("ret=%d\n", ret);
         return 0;
}

使用vs2017调试,查看反汇编

2.1 main函数的调用

main在调用之前如图:

2.2 sum函数的调用

sum函数在调用之前

sum函数内的内存分布

在上述的汇编码中我们可以看到在函数开始的时候,习惯上以这么l两段代码开始

push ebp  
 
mov ebp,esp

按照字面上理解,上面两句话的意思是将ebp推入栈中,之后让ebp等于esp

在函数调用之前,将调用者的函数(caller)的ebp存入栈,以便于在执行完毕后恢复现场;

下一步,sum函数必须为它的局部变量分配空间,同时,也必须为它可能用到的一些临时变量分配空间;

 sub esp, 0cch; // 减去的值根据程序而定;

之后会根据情况看是否保存某些特定的寄存器(EBX,ESI和EDI);

而ebp的值会保持固定,局部变量和临时存储则都可以通过基准指针ebp加偏移量找到;

在函数执行完毕,控制流返回到调用者的函数(caller)之前会进行下述操作

所谓有始有终,这是会还原上面保存的寄存器值,之后还原esp的值(上一个函数调用之前的esp被保存在固定的ebp中)与ebp值。这一过程被称为还原现场之后通过ret返回上一个函数。

参考:

https://blog.csdn.net/u011822516/article/details/20001765

https://blog.csdn.net/u012218309/article/details/81669227

https://www.cnblogs.com/sddai/p/9762968.html

https://blog.csdn.net/weixin_42572273/article/details/104611337

到此这篇关于深入理解C++函数栈帧 的文章就介绍到这了,更多相关C++函数栈帧 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C语言实现纸牌计算24点小游戏

    C语言实现纸牌计算24点小游戏

    这篇文章主要为大家详细介绍了C语言实现纸牌计算24点小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-10-10
  • C++中COM组件初始化方法实例分析

    C++中COM组件初始化方法实例分析

    这篇文章主要介绍了C++中COM组件初始化方法,涉及C++中COM组件的使用技巧,需要的朋友可以参考下
    2015-05-05
  • C++条件及循环语句的综合运用实例

    C++条件及循环语句的综合运用实例

    这篇文章主要介绍了C++条件及循环语句的综合运用实例,能够帮助C++初学者更好地掌握C++的逻辑语句用法,需要的朋友可以参考下
    2015-09-09
  • C++双目运算符+=的重载详解

    C++双目运算符+=的重载详解

    这篇文章主要介绍了详解C++编程中的双目运算符重载,是C++入门学习中的基础知识,需要的朋友可以参考下,希望能够给你带来帮助
    2021-09-09
  • C语言中进程间通讯的方式详解

    C语言中进程间通讯的方式详解

    这篇文章主要为大家详细介绍了C语言中几种进程间通讯的方式,文中的示例代码讲解详细, 对我们学习或工作有一定的借鉴价值,需要的可以参考一下
    2022-08-08
  • C++实现下载的代码

    C++实现下载的代码

    这篇文章主要介绍了C++实现下载的代码,以下载百度图片为例较为完整的讲述了C++下载的具体实现方法,需要的朋友可以参考下
    2014-10-10
  • C语言实现三子棋游戏附注释

    C语言实现三子棋游戏附注释

    这篇文章主要为大家详细介绍了C语言实现三子棋游戏附注释,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-06-06
  • C++特殊成员详解

    C++特殊成员详解

    这篇文章主要为大家介绍了C++特殊成员,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助,希望能够给你带来帮助
    2021-11-11
  • C++泛型模板约束深入讲解

    C++泛型模板约束深入讲解

    C/C++ 作为 C# 语言的前置版本,ECMA工业化编程语言,自然是存在 “泛型模板约束” 的功能的,只是本文不以 C/C++ 20 新语法搞出来的 “requires” 关键字来实现,它很难用
    2022-09-09
  • C++ 静态成员的类内初始化详解及实例代码

    C++ 静态成员的类内初始化详解及实例代码

    这篇文章主要介绍了C++ 静态成员的类内初始化详解及实例代码的相关资料,需要的朋友可以参考下
    2017-02-02

最新评论