当前位置: 首页 > news >正文

Linux中进程地址空间

目录

    • 程序地址空间
    • 进程地址空间

程序地址空间

在这里插入图片描述
我们在学习时,通常会以这样一张表来表示程序布局,我们写的变量,对象,代码数据等就可以和这张图对应,从而区分不同数据的位置。

那么程序加载到内存后的内存布局是这个程序分布图吗?

int un_gval;
int init_gval=100;int main(int argc, char *argv[], char *env[])
{printf("code addr: %p\n", main);//代码区const char *str = "hello Linux";printf("read only char addr: %p\n", str);//字符常量区printf("init global value addr: %p\n", &init_gval);//已初始化全局数据区printf("uninit global value addr: %p\n", &un_gval);//未初始化全局数据区char *heap1 = (char*)malloc(100); printf("heap addr : %p\n", heap1);//堆区printf("stack addr : %p\n", &str);//栈区return 0;
}

在这里插入图片描述
可以看到结果的确是按照程序分布图划分区域的

在这里插入图片描述
更细分的话可以看到堆向上增长,栈向下增长

复习一个概念,栈整体向下增长,但局部是向上使用的,就像数组
int a[10] , 遍历数组时通常需要++
在这里插入图片描述
对于struct b{x,y,z} ,struct b obj 中的 &obj.x 、&obj.y 、&obj.z也是遵循这样的规则
在这里插入图片描述
对于一个变量a,用static修饰局部变量之后,其实是把它作为全局变量了,所以函数调用结束不会被释放
在这里插入图片描述
用fork演示一下

int g_val = 100;int main()
{pid_t id = fork();if (id == 0){//childint cnt = 5;while (1){printf("child, Pid: %d, Ppid: %d, g_val: %d, &g_val=%p\n", getpid(), getppid(), g_val, &g_val);sleep(1);if (cnt == 0){g_val = 200;printf("child change g_val: 100->200\n");}cnt--;}}else{//fatherwhile (1){printf("father, Pid: %d, Ppid: %d, g_val: %d, &g_val=%p\n", getpid(), getppid(), g_val, &g_val);sleep(1);}}sleep(100);return 0;
}

在这里插入图片描述
发现对于同一个g_val,同样的地址去读取,父子进程读出了不同的内容,那么我们在C/C++看到的地址,肯定不是物理地址。
我们平时遇到的地址,都是虚拟地址/线性地址

进程地址空间

物理地址,用户一概看不到,由OS统一管理
OS必须负责将 虚拟地址 转化成 物理地址
我们之前说的程序布局表准确来说是叫 进程地址空间

每一个进程运行之后,都会有一个进程地址空间的存在。
在这里插入图片描述
对于每一个进程,只要通过页表,就能用虚拟地址找到,定位到映射的物理地址,从而找到变量数据

创建出子进程之后,子进程要以父进程为模版拷贝PCB,除了自身pid,ppid,优先级等。然后拷贝进程地址空间,让自己的PCB指向自己的进程地址空间,再拷贝页表。
在这里插入图片描述
如果这个时候,子进程想修改g_val的值
每一个进程具有独立性,子进程读取之前会由操作系统 在物理内存中进行写时拷贝
在这里插入图片描述
这个进程地址空间,本质和进程的task_struct一样,是一个结构体
在这里插入图片描述

进程地址空间中的对不同区域划分,代码区、栈区,本质就是赋不同的值:long code_end = 40 ,long data_start =40

http://www.lryc.cn/news/613405.html

相关文章:

  • 2025 年华数杯全国大学生数学建模竞赛C 题 可调控生物节律的 LED 光源研究--完整成品、思路、模型、代码、结果分享
  • GPT-5即将来袭,AI新时代要变天?
  • Redis实战(8) -- 分布式锁Redission底层机制
  • LVDS系列24:Xilinx Ultrascale系ISERDESE3原语(二)
  • 【数据结构——并查集】
  • 批量获取亚马逊商品SKU商品规格调用流程
  • 哈勃网络计划大规模升级卫星以创建全球蓝牙层
  • 哈希表——指针数组与单向链表的结合
  • [Oracle] FLOOR()函数
  • 2025最新国内服务器可用docker源仓库地址大全(2025年8月更新)
  • 上海一家机器人IPO核心零部件依赖外购, 募投计划频繁修改引疑
  • 【Linux基础知识系列】第八十八篇 - 使用du命令分析文件和目录大小
  • 如何解决用阿里云效流水线持续集成部署Nuxt静态应用时流程卡住,进行不下去的问题
  • 硬盘哨兵pe版本 v25.70.6 中文免费版
  • openGauss3.10企业版单机部署(openEuler20.03 SP3)
  • RP2040下的I2S Slave Out,PIO状态机(四)
  • HMC1119LP4METR ADI亚德诺 高频功率放大器 MMIC集成电路IC
  • 自动化测试篇--BUG篇
  • Android-Kotlin基础(Jetpack④-Room)
  • RepoCoder:仓库级代码补全的迭代检索生成框架解析与应用前沿
  • 前缀和
  • 网卡名eth1、em1 、eno1、ens1 的区别
  • C++ vector 扩容时到底发生了什么?
  • 纯本地AI知识库搭建:DeepSeek-R1+AnythingLLM全流程
  • priority_queue的使用和模拟
  • Kotlin中String的==相等比较符
  • C语言sprintf、strcmp、strcpy、strcat函数详解:字符串操作的核心工具
  • 「日拱一码」045 机器学习-因果发现算法
  • 力扣238:除自身之外数组的乘积
  • LeetCode算法日记 - Day 4: 三数之和、四数之和