进程地址空间

进程地址空间是什么

进程地址空间不是物理地址,是一种虚拟地址,由操作系统提供。进程地址空间本质是进程看待内存的方式,抽象出来的一个概念,内核中用一个结构体mm_struct表示,这样每个进程都认为自己独占系统内存资源。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在进程控制块task_struct中有一个mm_struct结构体指针,指向一个mm_struct结构体,这个结构体里面完成对各个数据区域的划分,然后通过页表映射到物理内存上。
在这里插入图片描述

区域划分本质:将线性地址空间划分成为一个一个的area,[start, end]
虚拟地址本质:在[start, end]之间的各个地址叫做虚拟地址。
在这里插入图片描述

为什么要存在地址空间

1.如果进程直接访问物理内存,那么我们看到的地址就是物理地址。c语言中可以用指针访问地址,如果指针越界了,有可能直接访问到另一个进程的代码和数据,这样的话进程的独立性无法保证。 因为物理内存暴漏,有可能有恶意程序直接通过物理地址进行内存数据的篡改。所以虚拟地址存在的第一个意义是保护物理内存,不受任何进程的直接访问,这样操作系统就可以在虚拟到物理之间转化的时候方便进行合法性校验。
2.将内存管理和进程管理进行解耦
3.让每个进程以同样的方式来看待代码和数据。

验证地址空间的基本排布

#include<stdio.h>
#include<stdlib.h>

int g_unval;    // 未初始化全局变量
int g_val = 100; // 已初始化全局变量 

int main(int argc, char* argv[], char* env[])
{
    printf("code addr         : %p\n", main);
    /* 打印代码区地址,main函数是函数,属于代码的一部分
     * 所以main函数的地址能充当代码区的地址 */

    const char* p = "hello world";
    printf("read only         : %p\n", p);
    /* 打印字符常量区的地址,p在栈上开辟空间
     * p里面保存字符串常量的起始地址 */

    printf("global val        : %p\n", &g_val);
    /* 打印已初始化全局数据区地址 */
    printf("global uninit val : %p\n", &g_unval);
    /* 未初始化全局数据区 */

    char* q = (char*)malloc(10);
    printf("heap addr         : %p\n", q);
    /* 打印堆区地址,指针变量q在栈区,q指向堆区 */ 

    printf("stack addr        : %p\n", &p);
    printf("stack addr        : %p\n", &q);
    /* 打印栈区地址 */

    printf("args addr         : %p\n", argv[0]);
    printf("args addr         : %p\n", argv[argc-1]);
    /* 命令行参数地址 */

    printf("env addr          : %p\n", env[0]);
    /* 环境变量 */
    return 0;
}

在这里插入图片描述

  • 10
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

北川_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值