AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 555289
Accepted
Lance
Lance
Asked: 2019-12-03 19:34:50 +0800 CST2019-12-03 19:34:50 +0800 CST 2019-12-03 19:34:50 +0800 CST

Linux中的堆栈使用什么数据结构?

  • 772

我已经查看了几个地方,例如这里,但没有一个详细解释用于实现堆栈本身的结构(“任务”(进程/线程)存储其嵌套调用信息等的地方)。它是一个链表,还是一个数组,还是别的什么?我似乎找不到此信息,但从图表上看,它们总是将其显示为一个大内存块(虚拟内存),其中开头是堆,结尾是堆栈。但这是我们正在处理的虚拟内存,它周围有各种数据结构,例如分页。所以问题是在这一切之上堆栈的实现到底是什么?我不禁认为它一定是一个链表。

原因是,如果你有多个进程,每个进程都有自己的堆栈,这是如何实现的?

在这里,我们似乎到达了某个地方:

每个进程在内核中运行时都有自己的堆栈供使用;在当前内核中,该堆栈的大小为 8KB 或(在 64 位系统上)16KB 内存。堆栈位于直接映射的内核内存中,因此它必须在物理上是连续的。

process thread
  • 1 1 个回答
  • 663 Views

1 个回答

  • Voted
  1. Best Answer
    Paul_Pedant
    2019-12-05T16:56:33+08:002019-12-05T16:56:33+08:00

    堆栈实际上是一个数组——它在连续的内存中包含一堆单词,有一个重要的限制——它只能在一端增长和收缩(因此 FILO——先进后出),它也是 LIFO。

    与数组的重要区别也是:处理器堆栈在逻辑上分为帧,并且(与数组不同),每个帧的大小都可以与其他帧不同。

    每一帧都包含进行函数调用时需要存储的内容,包括:

    • 被调用函数跳转的返回地址,在调用函数中继续。
    • 存放任何返回值的空间。
    • 传递给被调用函数的每个参数的副本。
    • CPU 寄存器的副本,以便独立函数中的寄存器优化不会干扰。

    如果您想知道一个函数如何在递归的同时对每个级别的所有参数和局部变量使用相同的名称,答案是它们都具有相对于它们当前堆栈帧的地址。

    每个处理器架构的栈帧结构都有不同的定义,以采用最自然的存储方式。没有“Linux”堆栈——英特尔、AMD 和 Sparc 都有自己的定义。请记住,您可以下载本地编译器必须知道如何从您自己的代码中调用的预编译库。

    Stack 本身也是一种通用数据结构。例如,如果您正在解析允许嵌套块构造的 C 或 SQL 或 XML 等语言的源代码,那么在执行过程中自然而然地构建您所在的块的堆栈。您不会希望使用进程堆栈来执行此操作:它是您正在解析的具有块结构的东西,而不是您自己的需要递归的代码。

    每个进程的堆栈只是其用户进程内存的一部分。通常,用户地址空间从 -8MB 到 0 到(比如说)60MB。堆栈从 -16 开始并向下增长(越来越负)。编译器分配的全局和静态内存从 0 开始向上,任何堆分配都在固定内存之上增长。代码是分开的(出于保护原因)。将负地址范围映射到分页内存不会对虚拟存储系统造成任何损害。

    • 2

相关问题

  • 如何用名称而不是“ps”中的数字替换用户 ID 和组 ID?

  • 如何获取进程的补充组 ID?

  • `/proc/irq/.../spurious` 包含什么?

  • 父进程可以改变其子进程的环境吗?

  • “弹出”如何让进程关闭文件句柄?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve