那么问题来了:在 4GB 物理内存的机器上,申请 8G 内存会怎么样呢?

news/2024/10/31 5:35:36/

大家好,我是小林。

早上看到读者在群里讨论这些面试题:

其中,第一个问题「在 4GB 物理内存的机器上,申请 8G 内存会怎么样?」存在比较大的争议,有人说会申请失败,有的人说可以申请成功。

这个问题在没有前置条件下,就说出答案就是耍流氓。因为在 32 位操作系统和 64 位操作系统场景下,答案是不同的。

另外,我们还要看申请完 8G 内存后会不会被使用,会被使用是一种情况,不会被使用又是另外一种情况了。

所以,我们要分场景讨论。

正文

应用程序通过 malloc 函数申请内存的时候,实际上申请的是虚拟内存,此时并不会分配物理内存。

当应用程序读写了这块虚拟内存,CPU 就会去访问这个虚拟内存, 这时会发现这个虚拟内存没有映射到物理内存, CPU 就会产生缺页中断,进程会从用户态切换到内核态,并将缺页中断交给内核的 Page Fault Handler (缺页中断函数)处理。

缺页中断处理函数会看是否有空闲的物理内存:

  • 如果有,就直接分配物理内存,并建立虚拟内存与物理内存之间的映射关系。
  • 如果没有空闲的物理内存,那么内核就会开始进行回收内存的工作,如果回收内存工作结束后,空闲的物理内存仍然无法满足此次物理内存的申请,那么内核就会放最后的大招了触发 OOM (Out of Memory)机制。

32 位操作系统和 64 位操作系统的虚拟地址空间大小是不同的,在 Linux 操作系统中,虚拟地址空间的内部又被分为内核空间和用户空间两部分,如下所示:

通过这里可以看出:

  • 32 位系统的内核空间占用 1G,位于最高处,剩下的 3G 是用户空间;
  • 64 位系统的内核空间和用户空间都是 128T,分别占据整个内存空间的最高和最低处,剩下的中间部分是未定义的。

现在可以回答这个问题了:在 32 位操作系统、4GB 物理内存的机器上,申请 8GB 内存,会怎么样?

因为 32 位操作系统,进程最多只能申请 3 GB 大小的虚拟内存空间,所以进程申请 8GB 内存,在申请虚拟内存阶段就会失败(我手上没有 32 位操作系统测试,我估计失败的原因是 OOM)。

在 64 位操作系统、4GB 物理内存的机器上,申请 8G 内存,会怎么样?

64 位操作系统,进程可以使用 128 TB 大小的虚拟内存空间,所以进程申请 8GB 内存是没问题的,因为进程申请内存是申请虚拟内存,只要不读写这个虚拟内存,操作系统就不会分配物理内存。

我们可以简单做个测试,我的服务器是 64 位操作系统,但是物理内存只有 2 GB。

现在,我在机器上,申请 4 GB 内存,注意下面代码只是单纯分配了虚拟内存,并没有使用该虚拟内存:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <string.h>int main() {int ret;char* addr[4];printf("使用cat /proc/%d/maps查看内存分配\n",getpid());size_t s = 1024 * 1024 * 1024;int i = 0;for(i = 0; i < 4; ++i) {printf("alloc size = %d\n", s);addr[i] = (char*) malloc(s);printf("主线程调用malloc后,申请1gb大小得内存,此内存起始地址:0X%x\n", addr[i]);}getchar();return 0;
}

然后运行这个代码,可以看到,我的物理内存虽然只有 2GB,但是程序正常分配了 4GB 大小的虚拟内存

我们可以通过下面这条命令查看进程的虚拟内存大小:

# ps aux | grep alloc_4g
USER       PID %CPU %MEM    VSZ   RSS TTY      STAT START   TIME COMMAND
root      7797  0.0  0.0 4198540  352 pts/1    S+   16:58   0:00 ./alloc_4g

其中,VSZ 就代表进程使用的虚拟内存大小,RSS 代表进程使用的物理内存大小。可以看到,VSZ 大小为 4198540,也就是 4GB 的虚拟内存。

然后,我们改一下代码,在申请完虚拟内存后,通过 memset 函数使用这个虚拟内存,看看会发生什么。

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <string.h>int main() {int ret;char* addr[4];printf("使用cat /proc/%d/maps查看内存分配\n",getpid());size_t s = 1024 * 1024 * 1024;int i = 0;for(i = 0; i < 4; ++i) {printf("alloc size = %d\n", s);addr[i] = (char*) malloc(s);printf("主线程调用malloc后,申请1gb大小得内存,此内存起始地址:0X%x\n", addr[i]);//访问虚拟内存memset(addr[i], 0, s);}getchar();return 0;
}

运行结果:

可以看到,在申请了 2GB 虚拟内存后,然后马上使用了这块虚拟内存,由于这台机器的物理内存只有 2 GB,所以发生了 OOM

至此, 验证完成了。简单总结下:

  • 在 32 位操作系统,因为进程最大只能申请 3 GB 大小的虚拟内存,所以直接申请 8G 内存,会申请失败。
  • 在 64位 位操作系统,因为进程最大只能申请 128 TB 大小的虚拟内存,即使物理内存只有 4GB,申请 8G 内存也是没问题,因为申请的内存是虚拟内存,等这块虚拟内存被访问了,因为物理空间不够,就会发生 OOM。

好了,水完了!

原文链接:
https://mp.weixin.qq.com/s/LbtXW1NtG6VqEpsqqOL3Ig

作者:小林coding


http://www.ppmy.cn/news/332808.html

相关文章

海思3559av100平台 8GB LPDDR4内存适配 分配

1. 前言 使用两片4GB LPDDR4组成8GB使用 下面会说明如何适配内存配置; 以及分配使用 2. 查找内存配置表 SDK目录: software\Hi3559AV100R001C02SPC031\02.only for reference\uboot 如下找到合适的ddr参数表 3. 制作uboot镜像 (1)生成reg_info.bin 使用hiregbin工具生…

android手机8g内存够用嘛,手机内存选8G还是12G,行业老人告诉你,多数人都花冤枉钱了...

现在的安卓手机主流内存为8GB&#xff0c;高配可以选择12GB&#xff0c;更有甚至已经推出了16GB、18GB版本&#xff0c;但其实很多人选手机内存都错了&#xff0c;浪费了不少冤枉钱&#xff0c;所以下面就由行业老人来告诉你&#xff0c;手机内存应该怎么选(苹果用户请绕道)。 …

python编程8g的内存够么_内存8G够用吗?深入解读~

导读:买笔记本,预算不是那么充足,不少本本8G内存的能便宜几百块,初次入手本本,就好很想知道,这个8G是否足够使用呢? 就在我发文时刻,我默默把任务管理器打开了; 没错,我的内存是8G的,而且我有独立显卡2G,其中以上浏览器状态,GPU应用4%左右,如果有PS、剪切视频等,…

计算机内存只认4,为什么电脑的8GB内存只有7.45GB可用?

今天偶遇到这样一个问题&#xff0c;本来8GB内存的笔记本系统却只有7.45GB可用&#xff01;具体什么现象请看下图&#xff1a; 丢失了0.55GB内存 为此我还特意百度了一下&#xff0c;得到这样的结果&#xff1a;电脑少的内存被核心显卡占用&#xff01; 核心显卡和系统共享内存…

【漫画+图解】面试官让我用 2GB 内存给 20亿个整数(需要8GB内存)排序,我该咋整??

排序的时候我们可以选择快速排序或归并排序等算法。为了方便&#xff0c;我们把排序好的2G有序数据称之为有序子串吧。接着我们可以把两个小的有序子串合并成一个大的有序子串。 注意&#xff1a;读取的时候是每次读取一个int数&#xff0c;通过比较之后在输出。 按照这个方法…

2.2 搭建Spark开发环境

一、Spark开发环境准备工作 由于Spark仅仅是一种计算框架&#xff0c;不负责数据的存储和管理&#xff0c;因此&#xff0c;通常都会将Spark和Hadoop进行统一部署&#xff0c;由Hadoop中的HDFS、HBase等组件负责数据的存储管理&#xff0c;Spark负责数据计算。 安装Spark集群前…

数据可视化学习笔记:Python实现汽车品牌销售量矩形树图

引言 本文将介绍如何使用 Python 和 Pyecharts 库创建一个汽车品牌销售量的矩形树图。我们将使用 Pandas 读取 CSV 文件数据,然后对数据进行处理、封装,最后将数据可视化为矩形树图。 准备工作 首先,我们需要先安装好相关库: PandasPyecharts可以使用 pip 命令进行安装:…

自己制作一寸照片

用ps制作1寸照片 一&#xff0c;文件–打开一张照片。 二&#xff0c;点击裁剪工具 (调整为宽:2.5cm,高:3.5cm,分辨率:300像素&#xff0c;注高版本ps记住选择厘米像素),设置之后直接框选出需要剪切保留的位置&#xff08;使人物居正中&#xff09;&#xff0c; 然后按上面的“…