栈帧详解ebp、esp

article/2025/10/4 8:20:21

一. 理解栈帧

栈帧是什么,我们基本的理解是栈帧也叫活动记录过程,是编译器用来实现过程
函数调用的一种数据结构。通俗来说栈帧就时C语言函数在调用的过程中的调用原理,就是当我们执行一个函数操作的时候,它的内部是如何实现的呢。

二 .关于栈帧的背景知识

  1. 寄存器

第一个寄存器ebp,基址寄存器,也叫做栈底寄存器。

第二个寄存器esp,是栈顶寄存器。

第三个寄存器pc指针,也叫做程序计数器,它永远指向当前指令的下一条指令。

  1. 计算机运算的基本过程

取指令–分析指令–执行指令

但程序执行的过程中,pc指针指向下一个指令,那么当一个函数执行的时候,它会指向这个 函数,另外任何一个函数都有自己的ebp和esp,即是任何一个函数都有一个栈底和栈顶。但是在我们的cpu中,ebp和esp不可能有很多,可是我们的函数却可以很多,所以在函数执行的时候,ebp和esp都被新的函数覆盖掉了,那个原来的ebp和esp都应该保存,这样之后我们在执行完一个函数之后才可以回到上面一个函数。所以我们的ebp和esp始终指向当前函数的栈底和栈顶。

  1. 栈地址的生长方向

栈的生长方向是从高地址往低地址生长的,那么随着我们函数的调用的层层深入,我们的ebp和esp会越来越小。

  1. 程序地址空间

这里先画一个草图,让大家看一下一个程序在运行的时候需要哪些地址空间。
在这里插入图片描述

今天我们主要讨论的是栈区里面的使用原理,下面就是栈帧调用的详细解析

三. 栈帧调用原理

这里讨论函数调用的原理,我们主要画一个栈和代码区的一个图。下面的这幅图简明的说明了栈帧调用的机理。

在函数的最开始执行的应该是调用main函数,那么此时应该为main函数开辟一个存储空间,并且有一个ebp和一个esp指向这个存储空间的栈顶和栈底,另外还有pc指针指向代码区中的main函数中的下一条指令。这里再次说明一下,
栈里面是由高地址往低地址生长,请看下图,其中红色箭头代表main函数的指针,蓝色箭头代表指向fun函数的指针,这里我们假设首先执行main函数,然后执行fun函数,首先使用红色指针,然后使用蓝色指针。
在这里插入图片描述

这里说明了一个问题就是,当程序执行不同的函数时,我们的ebp、esp和pc指针是变化的,并且执向当前函数的栈帧区,因为我们执行完当前函数之后还希望返回原来的函数,那么我们在改变这三个指针之前一定要先保存这三个指针。上图只是说明我们要保存三个指针变量但是没有说明具体是如何保存的,还有就是函数中是要开辟参数的,我们的草图也没有说明参数的空间时如何开辟的。

接下来我们通过具体的代码来详细的说明一下,在编译器的内部是如何进行栈帧的变化的。首先看下面的一段代码

#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<windows.h>
int fun(int a, int b)
{
int c = 0xcccccccc;
return c;
}
int main()
{
int a = 0xaaaaaaaa; //这里赋值为0xaaaaaaaa的原因是,在一会的查看汇编时候,
//显示的都是16进制,这里这样赋值为了我们容易观察
int b = 0xbbbbbbbb;
int ret = fun(a, b);
printf("You shoud running here!\n");
system("pause");
return 0;
}

对于这段代码,我们可以按F10进入调试,然后选择菜单栏的调试,选择反汇编,然后一次次按F10知道定义a变量的时候,看下面的截图。

在这里插入图片描述

这里看红色框里面,实际上在汇编语言中,定义并初始化a,b变量的时候实际上是把一个值放入到一个内存当中去,中间用到了mov指令。那么我们此时也应该在主函数的栈帧区为a,b变量分别开辟空间,请看下图。
在这里插入图片描述

接下来我们基础回到我们的汇编代码,请看我给大家的截图
在这里插入图片描述

这里补充说明的一点,汇编代码中的call实际上就是调用函数的过程,但是我们还需要明白的一点就是,当我们调用这个函数的时候,数据应该是已经创建好的,所以我们这里在call指令调用这个函数的前面是一个传递参数的工作。这里解释一点就是eax,ebx…都是通用寄存器。第一个红框里面mov指令,把b变量放置在eax中,然后执行push指令,该指令的作用是把b变量放置在栈顶,第二个红色框中,把a变量放置在ecx中,然后push变量a入栈。push完成过后就形成了形参实例化的a和b的临时变量。从上面的实例化的过程中,我们不难发现一个问题就是,参数在实例化的时候的顺序是从右往左的。

现在我们再来看看栈帧图是如何压栈的。
在这里插入图片描述

这里的压栈,即汇编代码中的push,是首先把指针往下移动,然后把数据形参b放进栈顶,然后再使指针往下移动,然后再放入数据形参a到栈顶位置。

把a,b的形参压如栈顶之后,才真正开始执行call指令,call指令的作用是,第一个作用是把当前指令的下一条指令的地址压入栈中,当前指令使call指令,它的下一条指令是add,从上面的反汇编图中可以看到,这里我们还可以看到add的指令地址为01011449,那么这个地址就被压入栈中;call的第二个作用是跳转(jmp)到下一个函数的入口。我们按F11就可以进入到jmp这个函数中去,jmp的一个作用是修改pc指针。接下来按F10就可以执行我们的fun函数。

这个时候我们进入到了它的fun函数内部,继续查看它的反汇编代码,请看下面的反汇编的截图
在这里插入图片描述

第一句,push ebp即是把main函数的栈底指针压栈。接下来是mov ebp,esp它的作用就是把ebp指向当前函数的栈底。再下一条指令,sub esp,0CCh,这条指令作用是使esp往下移一段空间,这样做之后,我们的ebp和esp又指向了一个新的空间,这个空间就是我们的fun函数的栈帧空间。

再来看上图中的最下面的两个红色框中,这个时候fun函数已经执行完毕,mov esp,ebp,就是把ebp中的内容放在esp中去,那么此时的结果就是esp又指向了存放main函数的ebp的地址空间,请看黑色箭头。接下来又执行了pop ebp,就是恢复ebp的值,那么此时ebp又指向了main函数的最开始的ebp中去了,然后esp往上退回一个地址空间,那么此时esp就是main:ret,也就是刚刚的那个call指令的下一条指令add指令。接下来反汇编代码中最后一个指令ret,这个指令被集成了,我们无法看到,它的作用是继续pop,它的其中一个操作是pop pc,那么就是把当前esp中的内容放在了pc指针中,此时pc指针指向的就是add指令,接着再让esp退回一段地址空间,所以ret执行完毕之后,esp又指回了临时变量a。请看下面的栈帧图

在这里插入图片描述

接下来我们按F10这样又回到了主函数当中去了,那个此时执行了add指令,请看下面的反汇编截图
在这里插入图片描述

紧接着它做了下一步操作add esp,8,就是把esp中的内容加上8,加上8之后,esp刚好指向了main函数原来的esp,至此esp和ebp都回到了原来的main的栈帧结构中了。

请看最后一张图,简化后的图
在这里插入图片描述


http://chatgpt.dhexx.cn/article/D4eyiONl.shtml

相关文章

【汇编】esp寻址与ebp寻址

前言&#xff1a;本教程使用的工具是DTDEBUG&#xff0c;讲解的是32位汇编。 1、什么是esp寻址 顾名思义&#xff0c;使用esp这个栈顶指针寄存器去寻找变量对应的地址&#xff0c;就叫做esp寻址。 如下就是一个简单的esp寻址&#xff1a; 像这样&#xff0c;我们通过esp的偏移…

栈帧ebp,esp详解

栈帧%ebp,%esp详解 分类专栏&#xff1a; 汇编 首先应该明白&#xff0c;栈是从高地址向低地址延伸的。每个函数的每次调用&#xff0c;都有它自己独立的一个栈帧&#xff0c;这个栈帧中维持着所需要的各种信息。寄存器ebp指向当前的栈帧的底部&#xff08;高地址&#xff0…

对于ESP、EBP寄存器的理解

转载&#xff1a;https://mp.weixin.qq.com/s/Od9X-qnQ3WWyZiLIS4uPFg 函数调用是编程语言都有的概念&#xff0c;也许你听说过函数调用栈&#xff0c;但是大家都知道函数调用是如何完成的吗&#xff1f;我们为什么要了解这个过程: 对于程序运行机制中的数据结构和实现的了解&…

ebpf简介

目录 什么是eBPFeBPF架构eBPF优势eBPF相关工具 什么是eBPF eBPF 是什么呢&#xff1f; 从它的全称“扩展的伯克利数据包过滤器 (Extended Berkeley Packet Filter)” 来看&#xff0c;它是一种数据包过滤技术&#xff0c;是从 BPF (Berkeley Packet Filter) 技术扩展而来的。顾…

函数栈EIP、EBP、ESP寄存器的作用

这一篇文章咱们就来重新认识一下EIP、EBP、ESP这三个寄存器&#xff0c;寄存器又好几个&#xff0c;但是为什么我们要单独看这几个呢&#xff1f;因为在很多情况下我们在调试的时候最注意的就是这三个寄存器&#xff0c;其实这几个寄存器都是为“栈”而生&#xff0c;下面将结合…

你了解函数调用过程吗?

函数调用是编程语言都有的概念&#xff0c;也许你听说过函数调用栈&#xff0c;但是大家都知道函数调用是如何完成的吗&#xff1f;我们为什么要了解这个过程: 对于程序运行机制中的数据结构和实现的了解&#xff0c;对自己开发程序有着启发作用碰到一些疑难杂症的时候&#x…

ebp/栈帧/call stack

一.什么是ebp&#xff1f; &#xff08;1&#xff09;ESP&#xff1a;栈指针寄存器(extended stack pointer)&#xff0c;其内存放着一个指针&#xff0c;该指针永远指向系统栈最上面一个栈帧的栈顶。 &#xff08;2&#xff09;EBP&#xff1a;基址指针寄存器(extended base …

栈帧详解

一. 理解栈帧 栈帧是什么&#xff0c;我们基本的理解是栈帧是栈帧也叫过程 活动记录&#xff0c;是 编译器用来实现过程/ 函数调用的一种数据结构。通俗来说栈帧就时C语言函数在调用的过程中的调用原理&#xff0c;就是当我们执行一个函数操作的时候&#xff0c;它的内部是如何…

EBP 和 ESP 详解

基本概念&#xff1a; &#xff08;1&#xff09;ESP&#xff1a;栈指针寄存器(extended stack pointer)&#xff0c;其内存放着一个指针&#xff0c;该指针永远指向系统栈最上面一个栈帧的栈顶。 &#xff08;2&#xff09;EBP&#xff1a;基址指针寄存器(extended base poin…

基于生命周期的开发方法——迭代开发方法

迭代开发方法 上一篇原型方法只是一种需求验证的手段&#xff0c;如果将其思想运用到整个开发过程&#xff0c;使得每个阶段的任务经过反复多次&#xff0c;或者将分析、设计、实施的周期反复多次&#xff0c;通过一次次迭代&#xff0c;不断在原来的基础上完善和修正&#xf…

2.迭代开发的过程是怎么样的

敏捷开发系列文章目录 在讨论PO如何给团队讲好故事这个问题之前&#xff0c;先给大家了解一些基本的敏捷概念&#xff0c;然后讲讲我们敏捷团队构成与整个敏捷开发的过程。 当初敏捷老师讲课的时候就跟我们所过&#xff0c;敏捷没有什么具体的形式&#xff0c;每个敏捷团队可能…

RUP之动态结构:迭代开发

迭代过程一般分为四个阶段&#xff1a;初始、细化、构造和移交&#xff0c;简称为I,E,C和T。每个阶段以一个重要的里程碑(milestone)结束。 初始(Inception)阶段 确定最终产品的构想及其业务用例、并定义项目范围 初始阶段以生命周期目标(LCO)里程碑为结束点 细化(Elaborat…

开发模式(敏捷开发,瀑布式开发,螺旋型开发,迭代开发,devOps开发)

一. 敏捷开发 以人为核心、迭代、循序渐进的开发方式 简化文档&#xff0c;提取文档重点&#xff0c;主要在于人与人之间的沟通&#xff0c; 对开发产品进行迭代&#xff0c;最终完成开发。 迭代&#xff1a;迭代是指把一个复杂且开发周期很长的开发任务&#xff0c;分解为很…

敏捷开发-快速迭代

&#xff08;转自:http://blog.csdn.net/xiaoxian8023/article/details/8883791&#xff09; 今天跟大家分享的是“敏捷开发、快速迭代”。我们大都采用的是“瀑布开发模式”&#xff0c;有了问题&#xff0c;就得返工&#xff0c;虽然最终的产品会比较齐全完善&#xff0c;但是…

一文搞定软件过程模型——瀑布模型、增量式开发/增量开发与迭代开发的区别

软件开发比较经典的过程模型有&#xff1a; 瀑布模型&#xff1a;该模型将基本的过程活动、描述、开发、有效性验证和进化&#xff0c;看成是一些界限分明的独立的过程阶段&#xff0c;例如&#xff0c;需求描述阶段、软件设计阶段、实现阶段、测试阶段等。增量式开发&#xf…

软件迭代开发流程

如果对python自动化测试、web自动化、接口自动化、移动端自动化、大型互联网架构技术、面试经验交流等等感兴趣的老铁们&#xff0c;可以关注我。我会在公众号&#xff08;程序员阿沐&#xff09;/群里&#xff08;810119819&#xff09;不定期的发放免费的资料链接&#xff0c…

迭代开发流程

转载于:https://www.cnblogs.com/gispathfinder/p/8747869.html

git 迭代开发分支流程规范

前言 Git 的名称太响了&#xff0c;相信大家都听说过&#xff0c;鼎鼎大名的同性交友网站 GitHub 就是基于 Git 作为唯一的版本库格式进行托管的平台。本篇不详细介绍 Git 的使用&#xff0c;仅介绍基于 Git 的开发分支流程规范&#xff0c;需要对 Git有一定的了解。 简述 G…

迭代开发和增量开发

&#xff08;转自&#xff1a;http://blog.csdn.net/l12345678/article/details/5642851&#xff09; “迭代”和“增量”是敏捷软件开发中的两个重要概念。弄清楚“迭代”和“增量”以及其依据&#xff0c;我们就可以在实际的操作中有章法可循。 为什么要迭代&#xff1f; 我们…

转」最佳方案:迭代式开发

前言 Fred Brooks 在 25 年前就曾写到&#xff1a;“不要指望一次成功&#xff0c;无论如何你都要这样。” 敏捷开发&#xff0c;小步快跑&#xff0c;持续迭代&#xff0c;不断改进&#xff0c;产品升级。 在用例需要之前&#xff0c;不要添加数据成员 在代码之前编写测试 …