《C语言程序设计进阶教程》一2.3 调用栈

2.3 调用栈

2.3.1 返回位置

本文讲的是C语言程序设计进阶教程一2.3 调用栈,计算机是怎样使用栈内存的呢?考虑下面的代码片段:

函数f2在第10行调用了f1。在f1完成它的任务后,程序从f1之后的那一行继续运行f2。图2.2描述了程序的流程。

假设如图2.3所示,一个标记插在f1被调用处的正下方。这个标记告诉程序在f1结束之后它应该在哪里继续下去。这叫作“返回位置”,它的含义为在函数f1返回之后(即在f1完成其任务之后),程序应该从此处继续执行。

一个函数在它执行到返回声明处就结束了——在这条声明下的一切都要被略过。考虑下面的例子:

在这个函数中,如果第3行的条件是真的,那么此函数将会在第6行执行return。在这种情况下,第7行的任何内容都会被略过,程序将从返回位置继续。然而,如果第3行的条件是假的,那么函数将在第9行开始执行代码。注意在第9行不需要有一个else。当函数到达第11行的时候,return被执行,函数就停止了——第12行被略过了。这里,“略过”的意思是当程序运行时这部分代码不会被执行。虽然第7行和第12行不会被执行,但如果它们包含了语法错误,源代码也不会被编译。下面,让我们考虑3个函数:

函数f3在第15行调用f2,f2在第8行调用f1。当f1结束之后,程序从调用f1之后的那行继续执行(第9行)。当f2结束之后,程序从调用f2之后的那行继续执行(第16行)。程序是怎样知道在一个函数结束之后该从哪里继续呢?当f3调用f2时,与“第16行”等价的机器码被压入栈内存。图2.4显示了当运行这个程序时函数调用的流程。

假设每个函数调用之后的那行被标记为一个返回位置(RL),如图2.5所示。本书使用行编号作为返回位置。本书中的调用栈是一个简化了的概念化模型,不反映任何具体型号的处理器。真正的处理器使用程序计数器而非行编号。

为什么栈内存“后入先出”的原则这么重要呢?栈内存存储着函数调用的倒序。因此程序才会知道它应该在f1结束之后从RL B而非RL A继续。程序使用栈内存来记住返回位置。栈内存也叫作调用栈,每一个C程序都由它来控制其函数执行的流程。几乎所有的计算机编程语言都采取这个方案。
我们的三函数程序执行时,调用栈可能会显示如下信息:当f3调用f2时,调用f2后的行编号(RL A)被压入栈内存。

当f2调用f1时,调用f1后的行编号(RL B)被压入栈内存。

当f1结束之后,行编号9就会出栈,程序在此行编号(9)处继续。调用栈现在有行编号16。

当f2结束之后,行编号被弹出,程序在此行编号(16)处继续。程序员不需要担心标记返回位置的问题,编译器会负责插入合适的代码来完成这件事。
知道为什么栈必须存储返回位置是有意义的。考虑这个例子:

函数f1在两个不同的位置(第8行和第11行)被调用。当f1在第8行被第一次调用时,程序在f1结束后从第9行(RL A)继续。当f1在第11行被第2次调用时,程序在f1结束后从第12行(RL B)继续。调用栈是一个管理这些返回地址的简单的方案,因为相同的函数(f1)可以在不同的地方被调用,必须安排一些返回地址来追踪将要执行的下一行代码。
调用栈的规则可以归结为如下几条:
当一个函数被调用时,这条调用之后的行编号就被压入调用栈。这个行编号就是“返回位置”(RL)。这是在被调用函数结束(即返回)之后程序继续执行的地方。
如果相同的函数在不同行处被调用,那么每个调用都有一个相应的返回位置(每个函数调用之后的那行)。
当一个函数结束之后,程序将从存储在调用栈顶部的行编号处继续。调用栈顶部的内容就会被弹出。
原文标题:C语言程序设计进阶教程一2.3 调用栈

时间: 2024-09-24 13:08:55

《C语言程序设计进阶教程》一2.3 调用栈的相关文章

《C语言程序设计进阶教程》一导读

前 言 为什么要写这本书 本文讲的是C语言程序设计进阶教程一导读,市面上有成百上千种关于编程的书籍,其中有很多都是关于C语言编程的,那么为什么我还要写这本书呢?为什么建议你花时间读它呢?这本书跟其他书有什么不同呢?跟很多作者一样,我写这本书是因为我觉得有必要,觉得这本书中的方法比其他书中的更好. 我将现在已有的关于编程的书分为两类:入门和进阶.入门类书是给初学者写的,一般都假设读者没有编程基础,所以主要是介绍基本的概念.通常以"Hello World!"程序开始,也就是将"H

《C语言程序设计进阶教程》一2.1 值和地址

第2章 栈 内 存 2.1 值和地址 本文讲的是C语言程序设计进阶教程一2.1 值和地址,在计算机中,程序和数据必须保存在叫作存储器(storage)的地方.没有存储器,计算机就没有可以计算的东西.存储器可以分为易失的和非易失的.易失的存储器需要供电,它只有在计算机开机的时候才能保存数据.易失的存储器通常叫作"内存".非易失存储器可以在计算机关机或者重启的时候保存数据,例如,闪存或者硬盘.闪存也叫作固态硬盘或SSD.现在一个典型的笔记本电脑会有几个GB的内存.G代表着"gig

《C语言程序设计进阶教程》一2.3.2 函数实参

2.3.2 函数实参 本文讲的是C语言程序设计进阶教程一2.3.2 函数实参,为了理解函数实参,我们必须对目前见到的十分简单的例子进行详细叙述.首先,大多数函数引入输入实参,具有返回值.<韦氏词典>将argument(实参)定义为"可以决定一个函数的值的自变量中的一个".对于一个数学函数,例如f(x.y.z),变量x.y.z就是函数f的实参.在C程序中,函数有相似的语法.考虑下面的例子: 输入a.b.c是f1的实参.当f1被调用的时候,f2必须提供3个实参,而且这些信息被压

《C语言程序设计进阶教程》一1.2 重定向输出

1.2 重定向输出 本文讲的是C语言程序设计进阶教程一1.2 重定向输出,printf函数可能是人们写C程序时最先了解的几个函数之一.著名的"Hello World!"程序经常被用作给初学者的例子.在这个计算机程序中,文本被打印至终端.然而在一些场合中,从程序中重定向信息并将其保存至一个文件中是很有用的.下面是几个可能用到的情境:一个程序打印得过多过快,电脑屏幕不能显示出打印的所有内容.你不想在程序运行时等待着,相反,你想稍后再去看上面的信息.有时,检查程序再次运行时的输出是否与之前相

《C语言程序设计进阶教程》一3.2 常见错误

3.2 常见错误 本文讲的是C语言程序设计进阶教程一3.2 常见错误,这里是一系列我所见过我的学生编写程序中的常见错误(有时甚至是我自己也会犯的).很多学生向我保证他们再也不会犯这些错误.事实上是人们还是会犯这些错误,而且比他们想象中的要更经常.这一节只考虑编程错误,而非设计错误.设计上的错误需要一本另外的关于设计软件方面的书来讲述. 原文标题:C语言程序设计进阶教程一3.2 常见错误

《C语言程序设计进阶教程》一2.5 习题

2.5 习题 本文讲的是C语言程序设计进阶教程一2.5 习题,本书有两种类型的作业:习题和编程作业.习题是不需要编写程序的问题--它们是"在纸上写"的问题.编程作业,显然是需要在电脑上完成的. 理解栈内存对于程序员而言是最必需的技能之一.如果你想要理解C程序(和许多其他编程语言),那么对栈内存的扎实理解是必要的. 原文标题:C语言程序设计进阶教程一2.5 习题

《C语言程序设计进阶教程》一3.2.2 错误数组下标

3.2.2 错误数组下标 本文讲的是C语言程序设计进阶教程一3.2.2 错误数组下标,对于一个有n个元素的数组,有效的下标是0,1,2, -, n-1,而n是一个无效的下标.当一个程序有着错误的下标时,这个程序就可能在一些情况下工作,而在其他的情况下崩溃.你不会想去编写一个靠运气工作的程序. 3.2.3 错误数据类型 你能骑自行车.你也可以用钢笔写字.你却不能骑钢笔,也不能用自行车写字.在程序中,类型指定功能.你需要理解并正确地使用类型.编程语言的趋势是让数据类型更有限制性,防止程序员犯意外的错

《C语言程序设计进阶教程》一3.1.3 编程后

3.1.3 编程后 本文讲的是C语言程序设计进阶教程一3.1.3 编程后,在你认为已经完成编程之后,读一读你的程序.检查有没有下面说的那些常见的错误.不要依赖测试:测试只会告诉你程序没有正常工作,它不会告诉你程序真的在正常运行.很可能测试用例并没有涵盖所有可能的方案.想要设计出涵盖所有可能方案的测试用例通常是很困难的.对于一个复杂的程序来说,涵盖所有可能方案一般是不可能的. 原文标题:C语言程序设计进阶教程一3.1.3 编程后

《C语言程序设计进阶教程》一3.2.1 未初始化变量

3.2.1 未初始化变量 本文讲的是C语言程序设计进阶教程一3.2.1 未初始化变量,一个常见的错误就是未初始化变量.一些学生认为所有的变量都会自动地初始化为0,这是错误的.未初始化的变量会储存着未占用值.这个值可能是0,但这是不一定的.这个类型的错误是很难通过测试来发现的.有时,这个值可能碰巧是0,这会让你认为程序是正确的.当值不为0时,程序就会有问题.一些学生认为初始化变量会使程序变慢--然而,这些纳秒级的延迟是可以忽略的.让你的程序慢几纳秒总比花几个小时去调试要好. 原文标题:C语言程序设

《C语言程序设计进阶教程》一2.3.4 值地址

2.3.4 值地址 本文讲的是C语言程序设计进阶教程一2.3.4 值地址,目前为止,我们所有函数的返回类型都是void,即函数什么都没有返回.函数可以返回值.考虑下面的例子: 局部变量u在f2中,所以它是在f2的栈帧中.u的值是未被定义的,因为它还未被赋值.记住C不初始化变量,所以未初始化的变量可以存储任何值(即未占用).f2的栈帧包含值还未被定义的变量u. u的地址是在f1被调用之前存储在调用栈的.这个地址叫作值地址,因为它是函数f1存储返回值的地址.因此,当f1的栈帧建立之后,就要为值地址再