Linux进程地址空间

文章通过一个C语言程序示例解释了进程地址空间的概念,指出我们在调试程序时看到的地址是虚拟地址而非真实的物理地址。每个进程都有自己独立的地址空间,使得它们能独占内存。当子进程修改全局变量时,由于写时拷贝机制,实际的物理地址会改变,但虚拟地址保持不变,从而保证了进程间的独立性和数据一致性。
摘要由CSDN通过智能技术生成


我们经常会调试程序,通过监视窗口能够看到每条指令、变量的地址、跳转等情况,但是我们看到的地址并不是真实的内存里的地址,而是假的!

举例子证实地址是假的!

#include <stdio.h>
#include <unistd.h>

int g_val = 10;

int main()
{
    pid_t id = fork();
    if(id == 0)
    {
        //Child
        int cnt = 5;
        while(1)
        {
            printf("I am child, pid = %d, ppid = %d, g_val = %d, &g_val = %p\n", getpid(), getppid(), g_val, &g_val);    
            --cnt;
            if(cnt == 0)
            {
                g_val = 30;
            }
            sleep(1);
        }   
    }
    //Parent
    while(1)
    {
        printf("I am parent, pid = %d, ppid = %d, g_val = %d, &g_val = %p\n", getpid(), getppid(), g_val, &g_val);    
        sleep(1);
    }   
    
    return 0;
}

截屏2023-01-03 20.41.44

可以看到,当子进程修改了g_val的值时,父子进程分别打印10和30,但是这两个的地址是一样的,也就是说同一块物理内存输出了两个不同的数值,但这怎么可能呢,这也恰恰说明了我们看到的地址是假的!

理解进程地址空间

计算机运行时,内存中同时存在很多的进程,但每个进程都认为自己是独占内存的!在32位机器上,每个进程都认为这16G的内存全部属于自己,为了实现这一思想,于是每个进程PCB里都存在一个结构体指针struct mm_struct *mm,该指针指向的结构体就是进程地址空间。

该结构体里存在众多指针,将16G内存划分为不同区域:数据区、代码区、栈区、堆区等等。每两个指针锁定一块区域。

//linux内核源代码
unsigned long start_code, end_code, start_data, end_data;
unsigned long start_brk, brk, start_stack;
unsigned long arg_start, arg_end, env_start, env_end;

image-20230103212132928

实际上,每个程序在编译完之后,程序内部就有虚拟地址(一般称为逻辑地址),而页表中虚拟和物理地址的映射关系就是程序内部的逻辑地址和物理内存的地址。也就是说程序内部也是有“地址空间”的,进程里也有地址空间,这两个地址空间一一对应。

为什么存在进程地址空间?

  1. CPU想执行代码,必须通过进程地址空间里的虚拟地址通过页表映射找到物理地址,从而执行代码。而页表在中间起到了警察的作用,如果进程里有非法的操作(越界),则页表会直接驳回操作。
  2. 进程地址空间的存在,更方便的使进程和进程之间的数据代码解耦,保证了进程的独立性。
  3. 让进程以统一的视角来看待进程对应的代码和数据等各个区域,方便编译器以统一的视角编译代码。

解释一开始的例子

fork创建子进程时,子进程以父进程为模版,拷贝了PCB,但并没有拷贝程序的数据和代码,所以此时父子进程的两个PCB都指向同一个程序,而当子进程对g_val进行修改时,发生写时拷贝,产生变化的变量会在内存中重新拷贝一份,子进程对于g_val指向新拷贝的,父进程仍旧指向旧的,但是此时只是子进程指向的g_val的物理内存地址变化了,子进程页表中的虚拟地址并没有变化,所以会导致两个进程的虚拟地址相同,但虚拟地址映射的物理地址不同。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

云朵c

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

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

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

打赏作者

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

抵扣说明:

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

余额充值