函数栈桢的创建与销毁@内功修炼

2023-12-26 20:20

本文主要是介绍函数栈桢的创建与销毁@内功修炼,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

引:

本文将解决你可能遇到的如下困惑:

  • 局部变量是怎样创建的?
  • 为什么局部变量不初始化时的值是随机值?
  • 函数是怎样传参的?传参顺序如何?
  • 形参与实参是什么关系?
  • 函数调用是怎么做的?
  • 函数调用后怎样返回的?

学会了函数栈桢的创建与销毁,其实就是修炼了自己的内功,也能搞懂后期很多知识。

本文时使用的环境是vs2013,注意不要使用太高级的编译器,越高级的编译器越不容易学习和观察。同时,不同的编译器下,函数调用中栈桢的创建也是略有差异的,具体细节取决于编译器的实现。

🍓嘿嘿:初次修炼内功在四个月之前,如上文所说,最近学习的东西又需要好好理解这部分内容,然而我功力渐退,于是有了这篇文章,作为再次修炼的结果。整个“过程”让人不禁感叹精妙!

正文开始

目录

  • 1. 知识铺垫
  • 2. 函数栈桢的创建
  • 3. 函数栈桢的销毁

1. 知识铺垫

为了观察函数栈桢的创建与销毁,这里采用最最简单的代码,并将其拆分的足够详细,便于观察:

#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);return 0;
}

知识铺垫与了解函数栈桢的大致轮廓:(细节后面聊)
在这里插入图片描述
我们知道,调用一个函数,都要在栈区上为它分配空间。
对于上面的一段代码 ——
在这里插入图片描述
这里我们需要知道,在vs2013中,main函数也是被其他函数调用的 ——可以看到,*main*函数是在

2. 函数栈桢的创建

下面将研究这个调用过程,大家跟着思路,逐步解答文章开头的疑问。
建议自己动手,F10调试起来,右击转到反汇编,把监视和内存窗口都打开,其实很有意思,因为真的是太精妙了!

来吧!

一上来,我们可以看到espebp还在维护调用main函数的__tmainCRTStartup这个函数的栈桢,这就是为什么刚刚提了一嘴main函数也是被其他函数调用的——在这里插入图片描述
那么接下来的绿色框框这几条汇编指令,就是在main函数预开辟栈桢的过程——
(注:栈区空间是高地址向低地址使用的,在插图中即是从下向上使用)
在这里插入图片描述
我们来配合画图和监视窗口一条条看吧!

可以看到ebp的确压到栈顶了,并且esp也随之而动了——
在这里插入图片描述
下面是在为main函数预开辟了一段空间 ——
在这里插入图片描述
接下来,我们push,push,push压栈,这是为了什么,我们不用管,继续 ——
在这里插入图片描述
这个绿框框中的汇编指令意思是,edi开始向下把39h(ecx)这么多个doubleword(dword)的数据全部赋为CCCCCCCC(eax).

在内存监视窗口,我们也可以看到,从edi开始好大一块空间都被改成了cccccccc ——
在这里插入图片描述
以上就是为main函数建立栈桢的全过程,至此我们才开始执行C语言代码 ——
在这里插入图片描述
我们来看,是怎样为局部变量a分配空间的 ——
在这里插入图片描述
🍓 这就解释了为什么变量没有初始化时默认的是随机值,如果我创建变量a没有初始化为10,那么默认就是cccccccc。事实上,我们之前经常不小心打印出来的"烫烫烫烫"就是内存中的cccccccc。这就是为什么创建变量时最好同时初始化。

我们继续bc的创建 ——
在这里插入图片描述
至此,我们明白了局部变量是如何创建的——建立栈桢,分配空间,初始化的话会赋值。

我们继续阅读汇编指令,接下来,我们要调用函数 —— 在call调用函数之前,绿色框框里发生了什么?
在这里插入图片描述
对的,实际上这就是在“传参” ,有趣的是,我们还没有调用Add函数,就已经传参过去了,而且是先传的b后传的a,从右向左传的 ——
在这里插入图片描述
做好准备工作了,接下来按F11我们调用Add函数,继续读指令,

call指令,让我们跳转到它后面的地址。有趣的是,与此同时栈顶压入了call指令的下一条汇编指令的地址。这是做什么用的?

我们能想到,再按F11我们jmpAdd函数之后,会执行Add函数中的一系列汇编指令,然而,调用完之后,我们还要回来接续它的下一条指令执行,因此要记住它的地址——
在这里插入图片描述
我们进入Add函数 :
那么最开始这几条指令就是在为Add函数开辟栈桢,可以看到,这的汇编指令和main函数一模儿一样,我们快进一下吧 。
在这里插入图片描述

但还是要注意,第一句指令push的是,此时正在维护main函数的栈底寄存器ebp。这个位置的记录,又为我们后面神奇事情的发生埋下了伏笔。(没关系,待会儿我们开始销毁的时候就能实实在在的感受它的作用了)

来吧,看快进结果 ——
在这里插入图片描述
建立好Add函数栈桢就是这样滴 ——
在这里插入图片描述
接下来,在Add函数栈桢中,我们同样为局部变量z分配了空间并初始化,再接下来我们就要计算啦 ——
在这里插入图片描述
那么z = x + y;是怎样计算的呢?
很有意思的是,我并没有在Add函数中创建形参,而是在调用Add函数之前就进行了压栈,在需要用它们的时候又通过指针的偏移回去找了压栈这段空间 ——
在这里插入图片描述
🍓形参是实参的一份临时拷贝这句话也是千真万确,它们的空间是独立的,形参的改变不会影响实参。
and呃 ——
在这里插入图片描述
🍓计算之后,返回值是如何带回来的呢?
从汇编指令可以看到Add函数栈桢马上就要开始销毁了,我们先把返回值放在寄存器中,等到回到调用它的这个函数,再把它赋给局部变量 ,这样函数销毁了也没有关系 ——
在这里插入图片描述
好嘞~ 我们接着看销毁过程吧!

3. 函数栈桢的销毁

上来就是三连pop,栈顶指针esp随之而动 ——
在这里插入图片描述
and then继续执行指令, Add函数栈桢被回收了,like this 哈哈——
在这里插入图片描述
接下来,神奇的事情就要发生了!
读下一条指令的意思是,pop弹出栈顶数据存入ebp,而此时栈顶元素是什么?

就是我们当初记录的main函数的栈底寄存器ebp呀!看呐!一切是如此的精妙、顺理成章!(完了,我写激动了)这样做是因为,随着函数栈桢的销毁,我们要找到它的栈顶是容易的,而它的栈底我却不记得了,因此要记录栈底,此时esp,ebp继续维护main函数栈桢

继续读指令,ret让我们返回栈顶的地址,即当初call的下一条指令的位置 。
一切是如此的精妙,这就再次解释了上文我们为什么要记录这个地址了,就是为了调用完函数之后,还能找回来 ——
在这里插入图片描述
调用完函数,回来,执行下一步指令,它在销毁形参
🍓在这里形参是如何销毁的?什么时候销毁的?我们也就清楚了 ——
在这里插入图片描述
就是这样 ——
在这里插入图片描述
继续读下一条指令,🍓返回值是怎样带回来的?
上文提到我们在函数栈桢销毁之前,把返回值存在eax寄存器中,在这条指令里,我们把eax的值赋给了局部变量——
在这里插入图片描述
就是这样——
在这里插入图片描述
至此,我们已经解答了开篇的所有问题,你还记得本文有几个草莓🍓吗?哈哈

我们再来回过头来看,这些问题已经在行文过程中都有了答案,在此再总结一下——

🍓局部变量是怎样创建的?
为函数分配好栈桢空间,栈桢空间初始化好,然后为局部变量分配空间。

🍓为什么局部变量不初始化时的值是随机值?
随机值是我放进去的,初始化即覆盖。

🍓函数是怎样传参的?传参顺序如何?
实际上我还没有调用时,我已经把这两个值从右向左传过去压栈,要使用时通过指针偏移量再找回。

🍓形参与实参是什么关系?
形参确实是实参的一份临时拷贝,是我在压栈时开辟的空间,值是相同的,但是空间是独立的。

🍓函数调用是怎么做的?
调用之前,我们就把call指令的下一条指令的地址压进去了,弹出ebp就能找到上一个函数的ebp

🍓 函数调用后怎样返回的?
返回值是通过寄存器帮我们带回来的,函数栈桢销毁了并没有影响。

文末碎碎念:回想上次自己调试不熟练,导致文章只开始了一点点就被其他事情冲走了,没有亲自上手把所发生的一切都记录下来的勇气。最近数据结构的练习,让我强迫自己调,忍住,别去找老师,几次调下来就自信了很多,于是本文的整个过程的记录就顺顺利利的完成了。

本文完

这篇关于函数栈桢的创建与销毁@内功修炼的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/540667

相关文章

Window Server创建2台服务器的故障转移群集的图文教程

《WindowServer创建2台服务器的故障转移群集的图文教程》本文主要介绍了在WindowsServer系统上创建一个包含两台成员服务器的故障转移群集,文中通过图文示例介绍的非常详细,对大家的... 目录一、 准备条件二、在ServerB安装故障转移群集三、在ServerC安装故障转移群集,操作与Ser

Window Server2016 AD域的创建的方法步骤

《WindowServer2016AD域的创建的方法步骤》本文主要介绍了WindowServer2016AD域的创建的方法步骤,文中通过图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、准备条件二、在ServerA服务器中常见AD域管理器:三、创建AD域,域地址为“test.ly”

Python在固定文件夹批量创建固定后缀的文件(方法详解)

《Python在固定文件夹批量创建固定后缀的文件(方法详解)》文章讲述了如何使用Python批量创建后缀为.md的文件夹,生成100个,代码中需要修改的路径、前缀和后缀名,并提供了注意事项和代码示例,... 目录1. python需求的任务2. Python代码的实现3. 代码修改的位置4. 运行结果5.

使用IntelliJ IDEA创建简单的Java Web项目完整步骤

《使用IntelliJIDEA创建简单的JavaWeb项目完整步骤》:本文主要介绍如何使用IntelliJIDEA创建一个简单的JavaWeb项目,实现登录、注册和查看用户列表功能,使用Se... 目录前置准备项目功能实现步骤1. 创建项目2. 配置 Tomcat3. 项目文件结构4. 创建数据库和表5.

使用SpringBoot创建一个RESTful API的详细步骤

《使用SpringBoot创建一个RESTfulAPI的详细步骤》使用Java的SpringBoot创建RESTfulAPI可以满足多种开发场景,它提供了快速开发、易于配置、可扩展、可维护的优点,尤... 目录一、创建 Spring Boot 项目二、创建控制器类(Controller Class)三、运行

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

手把手教你idea中创建一个javaweb(webapp)项目详细图文教程

《手把手教你idea中创建一个javaweb(webapp)项目详细图文教程》:本文主要介绍如何使用IntelliJIDEA创建一个Maven项目,并配置Tomcat服务器进行运行,过程包括创建... 1.启动idea2.创建项目模板点击项目-新建项目-选择maven,显示如下页面输入项目名称,选择

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>