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

简介: 本文讲的是C语言程序设计进阶教程一2.3 调用栈,本节书摘来华章计算机《C语言程序设计进阶教程》一书中的第2章,第2.3节, Intermediate C Programming[美] 陆永祥(Yung-Hsiang Lu) 著 徐东 译 译更多章节内容可以访问云栖社区“华章计算机”公众号查看。

2.3 调用栈

2.3.1 返回位置

本文讲的是C语言程序设计进阶教程一2.3 调用栈,计算机是怎样使用栈内存的呢?考虑下面的代码片段:
screenshot
函数f2在第10行调用了f1。在f1完成它的任务后,程序从f1之后的那一行继续运行f2。图2.2描述了程序的流程。

screenshot

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

screenshot

一个函数在它执行到返回声明处就结束了——在这条声明下的一切都要被略过。考虑下面的例子:
screenshot
在这个函数中,如果第3行的条件是真的,那么此函数将会在第6行执行return。在这种情况下,第7行的任何内容都会被略过,程序将从返回位置继续。然而,如果第3行的条件是假的,那么函数将在第9行开始执行代码。注意在第9行不需要有一个else。当函数到达第11行的时候,return被执行,函数就停止了——第12行被略过了。这里,“略过”的意思是当程序运行时这部分代码不会被执行。虽然第7行和第12行不会被执行,但如果它们包含了语法错误,源代码也不会被编译。下面,让我们考虑3个函数:
screenshot
函数f3在第15行调用f2,f2在第8行调用f1。当f1结束之后,程序从调用f1之后的那行继续执行(第9行)。当f2结束之后,程序从调用f2之后的那行继续执行(第16行)。程序是怎样知道在一个函数结束之后该从哪里继续呢?当f3调用f2时,与“第16行”等价的机器码被压入栈内存。图2.4显示了当运行这个程序时函数调用的流程。

screenshot

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

screenshot

为什么栈内存“后入先出”的原则这么重要呢?栈内存存储着函数调用的倒序。因此程序才会知道它应该在f1结束之后从RL B而非RL A继续。程序使用栈内存来记住返回位置。栈内存也叫作调用栈,每一个C程序都由它来控制其函数执行的流程。几乎所有的计算机编程语言都采取这个方案。
我们的三函数程序执行时,调用栈可能会显示如下信息:当f3调用f2时,调用f2后的行编号(RL A)被压入栈内存。
screenshot
当f2调用f1时,调用f1后的行编号(RL B)被压入栈内存。
screenshot
当f1结束之后,行编号9就会出栈,程序在此行编号(9)处继续。调用栈现在有行编号16。
screenshot
当f2结束之后,行编号被弹出,程序在此行编号(16)处继续。程序员不需要担心标记返回位置的问题,编译器会负责插入合适的代码来完成这件事。
知道为什么栈必须存储返回位置是有意义的。考虑这个例子:
screenshot
函数f1在两个不同的位置(第8行和第11行)被调用。当f1在第8行被第一次调用时,程序在f1结束后从第9行(RL A)继续。当f1在第11行被第2次调用时,程序在f1结束后从第12行(RL B)继续。调用栈是一个管理这些返回地址的简单的方案,因为相同的函数(f1)可以在不同的地方被调用,必须安排一些返回地址来追踪将要执行的下一行代码。
调用栈的规则可以归结为如下几条:
screenshot当一个函数被调用时,这条调用之后的行编号就被压入调用栈。这个行编号就是“返回位置”(RL)。这是在被调用函数结束(即返回)之后程序继续执行的地方。
screenshot如果相同的函数在不同行处被调用,那么每个调用都有一个相应的返回位置(每个函数调用之后的那行)。
screenshot当一个函数结束之后,程序将从存储在调用栈顶部的行编号处继续。调用栈顶部的内容就会被弹出。
原文标题:C语言程序设计进阶教程一2.3 调用栈

相关文章
|
5月前
|
C语言
指针进阶(C语言终)
指针进阶(C语言终)
|
1月前
|
C语言 开发者
C语言实现猜数字小游戏(详细教程)
C语言实现猜数字小游戏(详细教程)
|
1月前
|
编译器 C语言 C++
VSCode安装配置C语言(保姆级教程)
VSCode安装配置C语言(保姆级教程)
|
2月前
|
C语言
C语言程序设计核心详解 第四章&&第五章 选择结构程序设计&&循环结构程序设计
本章节介绍了C语言中的选择结构,包括关系表达式、逻辑表达式及其运算符的优先级,并通过示例详细解释了 `if` 语句的不同形式和 `switch` 语句的使用方法。此外,还概述了循环结构,包括 `while`、`do-while` 和 `for` 循环,并解释了 `break` 和 `continue` 控制语句的功能。最后,提供了两道例题以加深理解。
|
2月前
|
存储 C语言
C语言程序设计核心详解 第十章:位运算和c语言文件操作详解_文件操作函数
本文详细介绍了C语言中的位运算和文件操作。位运算包括按位与、或、异或、取反、左移和右移等六种运算符及其复合赋值运算符,每种运算符的功能和应用场景都有具体说明。文件操作部分则涵盖了文件的概念、分类、文件类型指针、文件的打开与关闭、读写操作及当前读写位置的调整等内容,提供了丰富的示例帮助理解。通过对本文的学习,读者可以全面掌握C语言中的位运算和文件处理技术。
|
2月前
|
存储 C语言
C语言程序设计核心详解 第七章 函数和预编译命令
本章介绍C语言中的函数定义与使用,以及预编译命令。主要内容包括函数的定义格式、调用方式和示例分析。C程序结构分为`main()`单框架或多子函数框架。函数不能嵌套定义但可互相调用。变量具有类型、作用范围和存储类别三种属性,其中作用范围分为局部和全局。预编译命令包括文件包含和宏定义,宏定义分为无参和带参两种形式。此外,还介绍了变量的存储类别及其特点。通过实例详细解析了函数调用过程及宏定义的应用。
|
2月前
|
C语言
C语言程序设计核心详解 第三章:顺序结构,printf(),scanf()详解
本章介绍顺序结构的基本框架及C语言的标准输入输出。程序从`main()`开始依次执行,框架包括输入、计算和输出三部分。重点讲解了`printf()`与`scanf()`函数:`printf()`用于格式化输出,支持多种占位符;`scanf()`用于格式化输入,需注意普通字符与占位符的区别。此外还介绍了`putchar()`和`getchar()`函数,分别用于输出和接收单个字符。
|
2月前
|
存储 算法 C语言
C语言程序设计核心详解 第一章:数制及转换与ASCII码
本专栏旨在夯实C语言基础,涵盖基础知识与进阶内容,助力解决自命题考试和考研问题,为数据结构与算法设计奠定坚实基础。内容包括数制及其转换、ASCII码、内存管理、机器码等,重点讲解二进制、八进制、十六进制的概念与转换方法,并介绍C语言的结构、数据类型和标识符规范。
|
2月前
|
存储 人工智能 C语言
C语言程序设计核心详解 第八章 指针超详细讲解_指针变量_二维数组指针_指向字符串指针
本文详细讲解了C语言中的指针,包括指针变量的定义与引用、指向数组及字符串的指针变量等。首先介绍了指针变量的基本概念和定义格式,随后通过多个示例展示了如何使用指针变量来操作普通变量、数组和字符串。文章还深入探讨了指向函数的指针变量以及指针数组的概念,并解释了空指针的意义和使用场景。通过丰富的代码示例和图形化展示,帮助读者更好地理解和掌握C语言中的指针知识。
|
2月前
|
存储 人工智能 C语言
C语言程序设计核心详解 第六章 数组_一维数组_二维数组_字符数组详解
本章介绍了C语言中的数组概念及应用。数组是一种存储同一类型数据的线性结构,通过下标访问元素。一维数组定义需指定长度,如`int a[10]`,并遵循命名规则。数组元素初始化可使用 `{}`,多余初值补0,少则随机。二维数组扩展了维度,定义形式为`int a[3][4]`,按行优先顺序存储。字符数组用于存储字符串,初始化时需添加结束符`\0`。此外,介绍了字符串处理函数,如`strcat()`、`strcpy()`、`strcmp()` 和 `strlen()`,用于拼接、复制、比较和计算字符串长度。