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

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

相关文章

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

C语言函数递归实际应用举例详解

《C语言函数递归实际应用举例详解》程序调用自身的编程技巧称为递归,递归做为一种算法在程序设计语言中广泛应用,:本文主要介绍C语言函数递归实际应用举例的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录前言一、递归的概念与思想二、递归的限制条件 三、递归的实际应用举例(一)求 n 的阶乘(二)顺序打印

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

Kotlin 作用域函数apply、let、run、with、also使用指南

《Kotlin作用域函数apply、let、run、with、also使用指南》在Kotlin开发中,作用域函数(ScopeFunctions)是一组能让代码更简洁、更函数式的高阶函数,本文将... 目录一、引言:为什么需要作用域函数?二、作用域函China编程数详解1. apply:对象配置的 “流式构建器”最

idea中创建新类时自动添加注释的实现

《idea中创建新类时自动添加注释的实现》在每次使用idea创建一个新类时,过了一段时间发现看不懂这个类是用来干嘛的,为了解决这个问题,我们可以设置在创建一个新类时自动添加注释,帮助我们理解这个类的用... 目录前言:详细操作:步骤一:点击上方的 文件(File),点击&nbmyHIgsp;设置(Setti

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

kotlin的函数forEach示例详解

《kotlin的函数forEach示例详解》在Kotlin中,forEach是一个高阶函数,用于遍历集合中的每个元素并对其执行指定的操作,它的核心特点是简洁、函数式,适用于需要遍历集合且无需返回值的场... 目录一、基本用法1️⃣ 遍历集合2️⃣ 遍历数组3️⃣ 遍历 Map二、与 for 循环的区别三、高

Spring 中使用反射创建 Bean 实例的几种方式

《Spring中使用反射创建Bean实例的几种方式》文章介绍了在Spring框架中如何使用反射来创建Bean实例,包括使用Class.newInstance()、Constructor.newI... 目录1. 使用 Class.newInstance() (仅限无参构造函数):2. 使用 Construc