发现有一些问题几乎是所有的新人都会遇到,而且也常因为缺乏一些基本的知识而无从下手。函数调用栈的内容就是其中之一。于是花点时间把以前写的内容整理出来。

程序在运行期间,内存中有一块区域,用来实现程序的函数调用机制。这块区域是一块LIFO的数据结构区域,我们可以叫函数栈(调用栈)。每个未退出的函数都会在函数栈中拥有一块数据区,我们叫函数的栈帧。函数的调用栈帧中,保存了相应的函数的一些重要信息:函数中使用的局部变量,函数的参数,另外还有一些维护函数栈所需要的数据,比如EBP指针,函数的返回地址。如下图。我们假设程序当前执行的函数是Z函数,那么在函数调用栈中就会存在类似像这样的结构(EBP所指向的其实是“父函数”的调用栈帧,如何做到的后面会解释):

编译器把C/C++代码编译成汇编指令时,会生成一系列(很有规则)的指令来支持函数调用的机制。当一个函数调用发生时(我们假设是Z函数内调用了A函数):

    1. 会执行零到多个PUSH指令(用于参数入栈),然后有执行一个CALL指令。CALL指令内部其实还暗含了一个将返回地址(即CALL指令下一条指令的地址)压栈的动作。之后,IP(instruction point)指向要跳转的指令的地址(CALL指令的目标地址,也就是下一个函数)

    2. 大部分的本地编译器都会在每个函数体之前插入类似如下指令:PUSH EBP; MOV EBP ESP;即,在程式执行到一个函数的真正函数体时,已有以下数据顺序入栈:参数,返回地址,EBP。(注意,EBP是如何作到指向上个函数调用栈帧的)

    3. 将栈顶指针进行上移,“空”出一块区域,用于临时地存放函数的局部变量。

这几步完成(也就是一个函数调用发生了),函数调用栈就会变成这个样子,函数调用结束的时候,相应的做“反向的动作”就可以了。

 

我们用一段非常简单的真实代码看来:

  1. int increase(int a) {
  2.     int temp = 4;
  3.     return a + 3;
  4. }
  5. int main(int argc, char* const argv[])
  6. {
  7.     int sum = increase(3);
  8.     return 0;
  9. }

在main函数中调用 increase 函数。用VS单步断点打开汇编模式,可以看到如下的代码

  1.     int sum = increase(3);
  2. 00D2561E  push        3  
  3. 00D25620  call        increase (0D2142Eh)  
  4. 00D25625  add         esp,4  
  5. 00D25628  mov         dword ptr [sum],eax  

对照前面的说明,我们可以看到,调用函数前有 push 指令先把函数参数压栈。之后才真正的call increase 。然后我们进入 increase 函数再看看函数体是什么样的。

  1. int increase(int a) {
  2. 000455C0  push        ebp  
  3. 000455C1  mov         ebp,esp  
  4. 000455C3  sub         esp,0CCh  
  5. 000455C9  push        ebx  
  6. 000455CA  push        esi  
  7. 000455CB  push        edi  
  8. 000455CC  lea         edi,[ebp-0CCh]  
  9. 000455D2  mov         ecx,33h  
  10. 000455D7  mov         eax,0CCCCCCCCh  
  11. 000455DC  rep stos    dword ptr es:[edi]  
  12.     int temp = 4;
  13. 000455DE  mov         dword ptr [temp],4  
  14.     return a + temp;
  15. 000455E5  mov         eax,dword ptr [a]  
  16. 000455E8  add         eax,dword ptr [temp]  
  17. }
  18. 000455EB  pop         edi  
  19. 000455EC  pop         esi  
  20. 000455ED  pop         ebx  
  21. 000455EE  mov         esp,ebp  
  22. 000455F0  pop         ebp  
  23. 000455F1  ret  

进入函数前,做的动作主要是保存各寄存器,注意“sub esp,0xcch”就是移动ESP,空出局部变量的“位置”,为什么只有一个局部变量,却生成了这么大块区域呢?

Stackoverflow上有解释:

This extra space is generated by the /Zi compile option. Which enables Edit + Continue. The extra space is available for local variables that you might add when you edit code while debugging.

You are also seeing the effect of /RTC, it initializes all local variables to 0xcccccccc so that it is easier to diagnose problems due to forgetting to initialize variables. Of course none of this code is generated in the default Release configuration settings.

从这段简单的代码中,我们可以知道函数调用大概是什么回事了。通过上面的内容,我们仔细体会下ESP和EBP两个寄存器的变化,也就下面向个指令

013D55C0  push        ebp      // 构建新的调用帧
013D55C1  mov         ebp, esp

013D55EE  mov         esp, ebp // 恢复到原来的调用帧
013D55F0  pop         ebp

再加上参数,返回地址,局部变量的入栈出栈,通过这样一种统一的、并不复杂代码生成模式和数据结构,可以应对任意复杂的函数调用情况,极其灵活。我一直觉得这是计算机科学中非常漂亮的一个创造,也是以简驭繁的一个经曲例子。

C开发基础--函数调用栈的更多相关文章

  1. 前端基础进阶(六):在chrome开发者工具中观察函数调用栈、作用域链与闭包

    在前端开发中,有一个非常重要的技能,叫做断点调试. 在chrome的开发者工具中,通过断点调试,我们能够非常方便的一步一步的观察JavaScript的执行过程,直观感知函数调用栈,作用域链,变量对象, ...

  2. 在chrome开发者工具中观察函数调用栈、作用域链与闭包

    在chrome开发者工具中观察函数调用栈.作用域链与闭包 在chrome的开发者工具中,通过断点调试,我们能够非常方便的一步一步的观察JavaScript的执行过程,直观感知函数调用栈,作用域链,变量 ...

  3. .NET基础拾遗(5)多线程开发基础

    Index : (1)类型语法.内存管理和垃圾回收基础 (2)面向对象的实现和异常的处理基础 (3)字符串.集合与流 (4)委托.事件.反射与特性 (5)多线程开发基础 (6)ADO.NET与数据库开 ...

  4. Android 开发基础及环境配置

    2011年买了第一部安卓操作系统的手机,当时势头正盛的HTC不可思议(incredible),当时的想法就是想学习下智能手机开发,但是由于各种原因,客观上是公司的项目太忙了,忙于项目管理.团队建设.客 ...

  5. Android 中调试手段 打印函数调用栈信息

    下面来简单介绍下 android 中的一种调试方法. 在 android 的 app 开发与调试中,经常需要用到打 Log 的方式来查看函数调用点. 这里介绍一种方法来打印当前栈中的函数调用关系 St ...

  6. Linux Debugging(一): 使用反汇编理解C++程序函数调用栈

    拿到CoreDump后,如果看到的地址都是????,那么基本上可以确定,程序的栈被破坏掉了.GDB也是使用函数的调用栈去还原"事故现场"的.因此理解函数调用栈,是使用GDB进行现场 ...

  7. go语言调度器源代码情景分析之四:函数调用栈

    本文是<go调度器源代码情景分析>系列 第一章 预备知识的第3小节. 什么是栈 栈是一种“后进先出”的数据结构,它相当于一个容器,当需要往容器里面添加元素时只能放在最上面的一个元素之上,需 ...

  8. Python运维开发基础09-函数基础【转】

    上节作业回顾 #!/usr/bin/env python3 # -*- coding:utf-8 -*- # author:Mr.chen # 实现简单的shell命令sed的替换功能 import ...

  9. Struts2开发基础

    Struts2开发基础 struts2采用拦截器的机制来处理用户的请求,使得业务逻辑控制器能够与ServletAPI完全脱离开. 1. Hello World! 配置web.xml <?xml ...

随机推荐

  1. silverlight ListBox 多列图片效果

    这个功能之前用wpf写过一次这次用Silverlight写一次 这两种写法上基本上没有太大的差别 这个Demo并不完美,只是给大家提供一个思路 源码:SilverLightListPricture.r ...

  2. 通过爬虫代理IP快速增加博客阅读量——亲测CSDN有效!

    写在前面 题目所说的并不是目的,主要是为了更详细的了解网站的反爬机制,如果真的想要提高博客的阅读量,优质的内容必不可少. 了解网站的反爬机制 一般网站从以下几个方面反爬虫: 1. 通过Headers反 ...

  3. CSS3小分队——进击的border-radius

    上一篇:<CSS float属性小解——”浮“生若水> 写在前面: ~~强势插入~~如果有想进一步了解float属性的小伙伴,可以猛戳上面的链接,<CSS float属性小解——”浮 ...

  4. Bootstrap系列 -- 20. 禁用状态

    Bootstrap框架的表单控件的禁用状态和普通的表单禁用状态实现方法是一样的,在相应的表单控件上添加属性“disabled” 在使用了“form-control”的表单控件中,样式设置了禁用表单背景 ...

  5. Zebra_Dialog 弹出层插件

    . Default dialog box, no custom settings. Click here to open. $.Zebra_Dialog('<strong>Zebra_Di ...

  6. 团队作业 -- beta版本

    下一阶段需要改进完善的功能 1界面布局 2方块颜色调整 下一阶段新增的功能 1分数排行榜 2撤销上一步操作 需要改进的团队分工 无. 按要求加上一起进行编码任务 需要改进的工具流程 使用github进 ...

  7. UML 几种关系的理解

    1,泛化关系 泛化关系的表现形式有3中,类A 集成类B  ,接口C  继承 接口D ,或者类E实现类F. 2,组合关系 组合关系描述的是整体与局部的关系,一个整体有很多部分组成,即整体包含的部分. 例 ...

  8. poj1470 LCA Tarjan

    比较直接的题目,入门一下. #include<map> #include<queue> #include<stack> #include<cmath> ...

  9. Gallery 图片画廊

    Gallery 图片画廊 学习网址http://amazeui.org/widgets/gallery?_ver=2.x

  10. 【CodeForces 227A】Where do I Turn?叉积

    题意 ABC的位置关系只有三种可能: 1.在一条直线上,输出TOWARDS A--B--C 2.AB 和BC垂直,B为直角顶点,AB左侧是C,输出LEFT C--B | A 3.AB 和BC垂直,B为 ...