操作系统概念的堆、栈不同于数据结构的堆、栈。

C 语言中,一切指针占 4 字节,这意味着指针指向 RAM 中的地址可以有 232 个,最小的地址是 0,最大的地址是 231 - 1。

(一)堆:

堆空间在内存中是一个字节的沙盒。

malloc()、free()、realloc() 是程序员使用软件,通过特定的启发式策略管理内存的。malloc() 不仅返回一块内存的基地址,它还留出额外的头部空间记录了分配出去的内存块的实际大小。

经过一些分配释放后,堆中的内存碎片化了,形成有大有小的空闲块。因此使用一个内存块的链表(存储下个空闲内存块的地址)来记录空闲的内存块,这个链表称为空闲表,方便 malloc() 等函数继续管理内存。

(二)栈:

当进行函数调用时,函数调用会强制地为局部变量申请空间,这些局部变量的内存就在 RAM 中被称为“栈”(stack segment)的子集中。

调用 main() 函数时会得到它的局部变量,它们 alive & active,当 main() 调用一个子函数时,main() 的变量就暂时被屏蔽掉了,无法访问;当继续调用子函数的子函数时,这一系列变量都被分配好空间,但只有在最底部的函数中的局部变量才是 alive 的,并可以通过变量名访问。

每当函数调用时,系统都会为其创建活动记录。

如调用以下 void fun(int para1, int para2) 函数,

void fun(int para1, int *para2) {
char a[];
short *b;
// 以下略
...
}

系统会形成这样的布局。

注意:

(1)函数形参从左到右,分别存放在内存的低地址到高地址,通俗地讲,第二个参数在第一个参数之上。

(2)位于函数形参和局部变量中间的部分,存储着调用函数的某些信息,告诉我们“哪块代码”调用了 fun(),这里称为 saved PC。

另一个递归调用阶乘函数的例子:

int fact(int n) {
if (n == ) return ;
return n * fact(n - );
}

我们用模拟汇编指令的方法写该 <fact> 函数:

 R1 = M[SP + ];    // 将 n 的值存放到寄存器 R1 中
BNE R1, , PC + ; // Branch分支指令,R1 和 0 Not Equal 时,跳转
RV = ;
RET; // RV 存放返回值 1,调用 RET 返回
R1 = M[SP + ];
R1 = R1 - ; // R1 = n - 1
SP = SP - ;
M[SP] = R1;
CALL <fact>; // 继续调用 <fact>,此时RV已经存放了 <fact> 返回的有意义的值
SP = SP + ;
R1 = M[SP + ];
RV = RV * R1;
RET;

其中,

SP(Stack Pointer)是一个特殊的寄存器,指向当前的相关活动记录的基地址,即当前栈的最低地址。

RV 存放返回值,调用 RET 时讲其返回给函数调用者。

本例假设系统中每一条指令都是 4 字节宽,因此第 2 行指令是 PC,则满足条件跳转到的 PC + 12 则是第 5 行。

首先,假设某函数,可能是 fact(4) 或其他函数,调用了 fact(3)。

如上个例子所示,函数调用时,为形参 3 创建的内存空间构成了当前活动记录的上半部分;SP 指向栈的最低地址,该处的 saved PC 保存了那个“某函数”的调用信息(哪行代码调用了 fact(3))。

将 n 的值 3 存放到寄存器 R1 中。

n(3) != 0,跳转到第 5 行。

准备 n - 1 的值 2。

创建下个要调用的函数 fact(2) 的活动记录的上半部分。

将 2 写入当前栈帧(stack frame)中。

调用 fact(2),函数 fact(2) 活动记录的 saved PC 保存了 fact(3) 函数指令第 10 行的地址。

进入 fact(2),当前 PC 执行到 fact(2) 的第 1 条指令:将 n 的值 2 保存到寄存器 R1 中。

同上,继续执行 fact(2),开始准备 n - 1 的值 1。

同上,创建 fact(1) 的活动记录的上半部分。

……

直到调用 fact(0)。

函数 fact(0) 返回值为 1。

开始层层返回。

最后返回值为 6,注意此时 SP 指向的是当前活动记录的基地址。

最终将 6 返回给调用这个原始调用 fact(3) 的函数。

到此,fact(3) 执行完毕。

以上

【OS】Heap & Stack的更多相关文章

  1. 【OS】NMON的简介和使用

    [OS]NMON的简介和使用 目前NMON已开源,以sourceforge为根据地,网址是http://nmon.sourceforge.net. 1. 目的 本文介绍操作系统监控工具Nmon的概念. ...

  2. 【LeetCode】栈 stack(共40题)

    [20]Valid Parentheses (2018年11月28日,复习, ko) 给了一个字符串判断是不是合法的括号配对. 题解:直接stack class Solution { public: ...

  3. 【LeetCode】Min Stack 解题报告

    [题目] Design a stack that supports push, pop, top, and retrieving the minimum element in constant tim ...

  4. 【DS】About Stack

    栈 一言以蔽之,就是后进的先出(LIFO). C语言实现代码: #include<stdio.h> #include<stdlib.h> typedef struct Stac ...

  5. 【leetcode】Min Stack -- python版

    题目描述: Design a stack that supports push, pop, top, and retrieving the minimum element in constant ti ...

  6. 【leetcode】Min Stack(easy)

    Design a stack that supports push, pop, top, and retrieving the minimum element in constant time. pu ...

  7. 【LeetCode225】 Implement Stack using Queues★

    1.题目 2.思路 3.java代码 import java.util.LinkedList; import java.util.Queue; public class MyStack { priva ...

  8. Python【OS】模块

    import osprint(os.getcwd())#取当前工作目录#os.chmod("day6-os模块.py",2)#给文件/目录加权限,对Windows的下面不好使(1. ...

  9. 【Leetcode】【Easy】Min Stack

    Design a stack that supports push, pop, top, and retrieving the minimum element in constant time. pu ...

随机推荐

  1. Http协议Status状态代码

    Http协议:Http协议(超文本传输协议)是一个基于请求与响应模式的.无状态的.应用层的协议,常基于TCP的连接方式,绝大多数的Web开发,都是构建在HTTP协议之上的Web应用http请求有三部分 ...

  2. spring datasource 使用 proxool

    XmlWebApplicationContext使用的xml配置如下: <?xml version="1.0" encoding="UTF-8"?> ...

  3. 常用Git命令清单。

    上期传送门:[清单]7个管理和优化网站资源的工具 下面是我整理的常用 Git 命令清单.几个专用名词的译名如下. Workspace:工作区 Index / Stage:暂存区 Repository: ...

  4. python操作wifi

    python连接wifi需要使用pywifi包,可以用pip install pywifi安装 1.导入包 import pywifi,time #保存包中写义的常量 from pywifi impo ...

  5. Mysql InnoDB 数据更新/删除导致锁表

    一. 如下对账表数据结构 create table t_cgw_ckjnl ( CNL_CODE ) default ' ' not null comment '通道编码', CNL_PLT_CD ) ...

  6. IdentityServer4授权和认证对接数据库

    接着上一篇讲:https://www.cnblogs.com/nsky/p/10352678.html 我们之前都是用in-men的方式把数据添加到内存了,目的是为了测试方便, 现在我们把所有配置都添 ...

  7. Qt 快捷键 复制当前行 向上复制 && 向下复制

    网上的答案不一,我的快捷键是默认的配置,未做过修改. 向前复制当前行: Ctrl + Alt + up (向上箭头) 向后复制当前行:Ctrl + Alt + down(向下箭头) 补充一个:Ctrl ...

  8. 二叉查找树ADT

    二叉查找树ADT 定义:是一个二叉树,其中每一个节点的值大于左子树的所有值而小于右子树的所有值 平衡二叉树:平衡是指一个二叉树的任何节点的深度均不得过深 AVL树 定义:是一个二叉查找树,每个节点的左 ...

  9. 灵雀云容器PaaS平台助力知名股份制银行金融科技革新

    互联网.科技和金融的碰撞给银行业带来巨大影响.IT技术起初是传统金融提升效率的工具和方法,随着新技术的演进,技术成为驱动变革的核心要素.Fintech金融科技以技术和数据为驱动,用创新的方法改变了金融 ...

  10. CentOS 7 搭建CA认证中心实现https取证

    CA认证中心简述 CA :CertificateAuthority的缩写,通常翻译成认证权威或者认证中心,主要用途是为用户发放数字证书 功能:证书发放.证书更新.证书撤销和证书验证. 作用:身份认证, ...