【Linux 系统标准 进程资源】Linux 创建一个最基本的进程所需的资源分析,以及线程资源与之的差异

简介: 【Linux 系统标准 进程资源】Linux 创建一个最基本的进程所需的资源分析,以及线程资源与之的差异

Linux 创建一个最基本的进程所需的资源分析

资源类型 描述 深度见解
内存资源 存储代码和数据 内存是进程存在的基础
CPU 时间 执行代码 CPU 时间决定了进程的执行速度
文件描述符 标识文件和网络连接 文件是进程与外界交互的一种方式
进程 ID 唯一标识一个进程 进程 ID 是进程的“身份证”
环境变量 存储配置信息 环境变量影响进程的行为

1. 内存资源

Linux 创建一个进程时,首先需要分配内存空间来存储进程的代码、数据、堆和栈。这些内存区域通常包括:

  • 代码段(Code Segment)
  • 数据段(Data Segment)
  • 堆(Heap)
  • 栈(Stack)

代码示例:

#include <unistd.h>
int main() {
    fork(); // 创建一个新进程
    return 0;
}

2. CPU 时间

进程创建后,需要 CPU 时间来执行其代码。这涉及到进程调度,即操作系统决定哪个进程应该使用 CPU。

3. 文件描述符

每个进程都有一组文件描述符,用于标识打开的文件和网络连接。

代码示例:

#include <fcntl.h>
int main() {
    int fd = open("file.txt", O_RDONLY); // 打开一个文件
    return 0;
}

4. 进程 ID 和父进程 ID

每个进程都有一个唯一的进程 ID(PID)和一个父进程 ID(PPID)。

5. 环境变量和命令行参数

进程还需要存储环境变量和命令行参数。

6. 内核数据结构

操作系统内核维护了一些数据结构,如进程表,以跟踪进程的状态。

源码角度:

在 Linux 源码中,进程创建主要通过 fork 系统调用实现,具体在 kernel/fork.c 文件的 do_fork 函数里。

7. 内存占用评估

7.1 基础内存需求

在 Linux 系统中,一个最基本的进程的内存占用主要由以下几个部分组成:

  1. 代码段(Code Segment): 存储程序的机器代码。
  2. 数据段(Data Segment): 存储全局变量和静态变量。
  3. 堆(Heap): 动态分配的内存空间。
  4. 栈(Stack): 存储局部变量和函数调用信息。
  5. 内核栈(Kernel Stack): 存储进程在内核态下的执行信息。
  6. 页表(Page Table): 存储虚拟地址到物理地址的映射信息。

7.2 内存占用的具体数值

具体的内存占用数值会因系统和编译器的不同而有所不同。一般来说,一个空的 C 程序在 64 位 Linux 系统上的内存占用大约如下:

  • 代码段:约 4KB
  • 数据段:约 1KB
  • 堆:初始通常为空,按需分配
  • 栈:约 8KB(可动态增长)
  • 内核栈:约 8KB
  • 页表:约 4KB

总计:约 25KB

源码角度:

在 Linux 源码中,进程的内存布局和分配主要在 mm(Memory Management)模块中定义,具体可以查看 mm/mmap.cmm/page_alloc.c 文件。

7.3 进程ID(PID)

进程ID(Process ID)通常是一个整数,用于唯一标识一个进程。在大多数Linux系统中,这是一个4字节(32位)的整数。因此,每个进程的PID会占用4字节的内存。

7.4 文件描述符

文件描述符(File Descriptor)是一个非负整数,用于标识一个打开的文件或其他I/O资源。文件描述符通常存储在进程的文件描述符表中,这是一个数组结构。每个文件描述符通常是一个4字节(32位)的整数。

假设一个进程打开了N个文件,那么文件描述符表的大小大致为 N * 4字节

7.5 总结

资源类型 内存占用 描述
代码段、数据段、堆和栈 几KB到几MB 存储程序的代码和数据
进程ID(PID) 4字节 唯一标识一个进程
文件描述符 N * 4字节 标识打开的文件和网络连接
内核数据结构 几十到几百字节 存储进程状态、调度信息等

这个表格总结了Linux创建一个最基本进程所需的内存资源。总体来说,至少需要几KB的内存。这还不包括其他类型的资源,如CPU时间、环境变量等。希望这次的总结能更准确地解答您的问题。

线程的资源消耗于进程的差异

线程和进程有以下主要区别在内存和资源占用方面:

  1. 共享资源:线程共享相同的地址空间、代码段、数据段和开放的文件等,因此不需要像进程那样为这些分配独立的内存。
  2. 栈空间:每个线程有自己的栈空间,这是它们主要的内存开销。
  3. 内核数据结构:线程通常有更少的内核数据结构,这意味着它们通常比进程更轻量级。
  4. 文件描述符:线程通常共享同一组文件描述符,而进程有自己独立的文件描述符。
  5. 寄存器状态:每个线程有自己的寄存器状态,但这通常比进程的寄存器状态要少。

总体来说,线程通常比进程更节省内存和其他资源,这也是为什么在需要高并发的应用中,线程通常更受欢迎。


资源类型 进程 线程 描述
代码段、数据段 独立 共享 存储程序的代码和数据
独立 共享 动态分配的内存空间
独立 独立 存储局部变量和函数调用信息
进程/线程 ID 4字节 4字节 唯一标识一个进程或线程
文件描述符 独立 共享 标识打开的文件和网络连接
内核数据结构 独立 较少 存储进程/线程状态、调度信息等
寄存器状态 独立 独立 存储CPU寄存器状态

从这个表格中,您可以看到:

  • 线程与进程共享代码段、数据段和堆,这减少了内存占用。
  • 线程有自己独立的栈和寄存器状态,但通常比进程更轻量级。
  • 线程共享文件描述符,而进程有自己的一套。

这样应该更清晰地展示了线程和进程在资源占用方面的差异。希望这次能解答您的疑问。

结语

在我们的编程学习之旅中,理解是我们迈向更高层次的重要一步。然而,掌握新技能、新理念,始终需要时间和坚持。从心理学的角度看,学习往往伴随着不断的试错和调整,这就像是我们的大脑在逐渐优化其解决问题的“算法”。

这就是为什么当我们遇到错误,我们应该将其视为学习和进步的机会,而不仅仅是困扰。通过理解和解决这些问题,我们不仅可以修复当前的代码,更可以提升我们的编程能力,防止在未来的项目中犯相同的错误。

我鼓励大家积极参与进来,不断提升自己的编程技术。无论你是初学者还是有经验的开发者,我希望我的博客能对你的学习之路有所帮助。如果你觉得这篇文章有用,不妨点击收藏,或者留下你的评论分享你的见解和经验,也欢迎你对我博客的内容提出建议和问题。每一次的点赞、评论、分享和关注都是对我的最大支持,也是对我持续分享和创作的动力。

目录
相关文章
|
5天前
|
Linux C语言 调度
|
5天前
|
Unix Linux 调度
linux线程与进程的区别及线程的优势
linux线程与进程的区别及线程的优势
|
5天前
|
存储 算法 Linux
【Linux】线程的内核级理解&&详谈页表以及虚拟地址到物理地址之间的转化
【Linux】线程的内核级理解&&详谈页表以及虚拟地址到物理地址之间的转化
|
5天前
|
存储 安全 Linux
【Linux】详解进程通信中信号量的本质&&同步和互斥的概念&&临界资源和临界区的概念
【Linux】详解进程通信中信号量的本质&&同步和互斥的概念&&临界资源和临界区的概念
|
5天前
|
Java
【Java多线程】分析线程加锁导致的死锁问题以及解决方案
【Java多线程】分析线程加锁导致的死锁问题以及解决方案
26 1
|
5天前
|
算法 Java Linux
【探索Linux】P.23(线程池 —— 简单模拟)
【探索Linux】P.23(线程池 —— 简单模拟)
11 0
|
5天前
|
存储 安全 Java
【探索Linux】P.21(多线程 | 线程同步 | 条件变量 | 线程安全)
【探索Linux】P.21(多线程 | 线程同步 | 条件变量 | 线程安全)
15 0
|
5天前
|
算法 安全 Linux
【探索Linux】P.20(多线程 | 线程互斥 | 互斥锁 | 死锁 | 资源饥饿)
【探索Linux】P.20(多线程 | 线程互斥 | 互斥锁 | 死锁 | 资源饥饿)
12 0