Linux下的内存溢出,其实是触发了OOM killer,你知道吗?
ahcoder 2025-01-20 10:35 9 浏览
OOM killer是linux内核在内存不足情况下的一种管理机制,当内核检测到系统物理内存不足时,就会通过OOM killer机制kill掉一些进程,kill进程的原则是通过使用一套启发式算法,它会计算所有进程的分数,然后选出那个分数最高的进程。进而kill掉,一般分数最高的进程占用的内存刚好是最大的。
那么为什么会突然出现内存不足的情况呢,这就要说说进程与内存的运行机制,默认情况下,Linux内核根据应用程序的要求分配内存,通常来说应用程序分配了内存但是并没有实际全部使用,为了提高性能,这部分没用的内存可以留作它用,这部分内存是属于每个进程的,内核直接回收利用的话比较麻烦,所以内核采用一种过度分配内存(over-commit memory)的办法来间接利用这部分 “空闲” 的内存,提高整体内存的使用效率。这个问题最类似的就是我们使用的宽带运营商了,比如现在电信宽带都承诺卖给用户的都是300Mb光纤带宽,而这实际上远远超出了他们的网络容量。他们赌的就是用户实际上并不会同时使用并用完分配给我们的带宽上限。
一般来说这样做没有问题,但当大多数应用程序都消耗完自己的内存的时候麻烦就来了,如果这些应用程序的内存需求加起来超出了物理内存(包括 swap)的容量,这就会导致系统可用内存迅速降低甚至不够用的情况,也就是没有内存页能够再分配给进程了。此时,内核必须kill掉一些进程才能腾出内存空间保障系统正常运行,于是,OOM killer机制被激活了,并通过启发式算法找出了要终结的进程。
理解了这个机制之后,我们就很容易理解了为啥JAVA进程躺着也能中枪了,因为它在系统上一般占用内存最多,所以如果Out of Memeory (OOM) 的话总,是不幸第一个被kill掉的应用。
OOM killer机制也是可配置的,我们可以通过一些内核参数来调整OOM killer的行为,避免系统在那里不停的kill进程。kill进程是根据每个进程打分的高低来决定的,root 权限的进程通常被认为很重要,不应该被轻易杀掉,所以打分的时候可以得到 3% 的优惠,而我们可以在用户空间通过操作每个进程的oom_adj内核参数来降低哪些进程被OOM killer选中kill掉的几率。比如,如果不想java进程被轻易杀掉的话,可以找到java运行的进程号后,调整oom_score_adj的值即可,例如,查看系统pid为30522的进程oom_score的值,可以执行如下操作:
[root@hadoopgateway ~]# cat /proc/30522/oom_score
22
例如要调整此进程的adj值为-16的话,可以执行如下操作:
[root@hadoopgateway ~]#echo -16 > /proc/30522/oom_score_adj
[root@hadoopgateway ~]# cat /proc/30522/oom_score
6
从这个操作过程中可以看到,在用户空间可以通过操作每个进程的oom_adj内核参数来调整进程的分数,而这个分数是通过oom_score这个内核参数看到的。
知道了oom_score_adj与oom_score的含义和设置后,可以写一个简单的脚本,查看下当前系统上oom_score分数最高(最容易被OOM Killer杀掉)的进程,脚本内容如下:
[root@hadoopgateway ~]# vi oom.sh
#!/bin/bash
for proc in $(find /proc -maxdepth 1 -regex '/proc/[0-9]+'); do
printf "%2d %5d %s\n" \
"$(cat $proc/oom_score)" \
"$(basename $proc)" \
"$(cat $proc/cmdline | tr '\0' ' ' | head -c 50)"
done 2>/dev/null | sort -nr | head -n 10
[root@hadoopgateway ~]# chmod 755 oom.sh
[root@hadoopgateway ~]# ./oom.sh
25 9571 /usr/java/default/bin/java -Xmx1000m -Djava.net.pr
18 22193 /usr/java/default/bin/java -Xmx256m -Djava.net.pre
16 21455 /usr/java/default/bin/java -Xmx256m -Djava.net.pre
15 26121 /usr/java/default/bin/java -Xmx256m -Djava.net.pre
13 17528 /usr/java/default/bin/java -Xmx256m -Djava.net.pre
12 6071 /usr/java/default/bin/java -Xmx256m -Djava.net.pre
12 18392 /usr/java/default/bin/java -Xmx256m -Djava.net.pre
1 933 /data/redis/bin/redis-server *:6379
1 9212 -bash
1 9210 sshd: root@pts/6
这个输出中,pid为9571的进程,oom_score分数最高,最容易被OOM Killer杀掉。
当然,我们也可以完全关闭OOM killer,但不推荐用在生产环境下关闭,执行如下操作:
[root@hadoopgateway ~]#sysctl -w vm.overcommit_memory=2
[root@hadoopgateway ~]# echo "vm.overcommit_memory=2" >> /etc/sysctl.conf
vm.overcommit_memory有0、1、2三个可选值,分别代表如下含义:
0:表示用户申请内存的时候,系统会判断剩余的内存多少,如果不够的话那么就会失败。
1: 用户申请内存的时候,系统不进行任何内存是否够用的检查,直到使用内存超过可用内存。
2: 用户一次申请的内存大小不允许超过可用内存的大小。
理解了OOM killer,就明白了为啥系统经常会出现内存溢出的现象了。好啦,今天就说到这里。
相关推荐
- ARM64内核内存布局图(ARM64内核内存布局图解)
-
ARM64架构处理器采用48位物理寻址机制,最大可以寻找到256TB的物理地址空间。对于目前的应用来说已经足够了,不需要扩展到64位的物理地址寻址。虚拟地址也同样最大支持48位支持,所以在处理器的架构...
- ARM64 linux 调试串口通信(ARM64 linux 调试串口通信实验报告)
-
ARM64linux调试串口通信随着国产机普及很多工作也转移到了新平台上,以前调试设备用的笔记本电脑也换成新国产ARM64架构的了。本文以绿联CM204USB-A转RJ45Console调试线...
- Gentoo Linux 终止对 Itanium IA-64 体系的支持
-
GentooLinux是最后几个继续维护Itanium(IA-64)架构构建的Linux发行版之一,但现在这些已停产的英特尔处理器正在逐步淘汰。由于Linux6.7内核放弃了对Itan...
- 如何检查 Linux 系统是 32 位还是 64 位?这9个命令查的又快又准!
-
在Linux系统中,位数(bit)通常指的是CPU架构的位宽,即CPU一次能够处理的数据量。32位系统和64位系统在内存寻址能力、计算性能和软件支持上存在显著差异:「32位系统」:...
- 调出好画面!带你玩转飞凌嵌入式AM62x开发板的显示接口
-
来源:飞凌嵌入式官网“显示”是嵌入式开发板最为重要的功能之一,能够支持更多种类、更高规格的显示接口,意味着它能够应对的使用场景也更加广泛。每一款嵌入式开发板在出厂前都会做屏幕调试,但在客户的实际项目开...
- 带你玩转AM62x开发板的显示接口——LVDS的显示和修改方式
-
此前小编已为大家介绍过OK6254-C开发板的RGB显示和修改方式,今天将继续为大家介绍OK6254-C开发板的LVDS显示和修改方式。话不多说,我们进入正题。1、LVDS接口规格飞凌嵌入式OK62...
- AM335x继任者?AM6254性能解析(am2361p)
-
飞凌嵌入式FET6254-C核心板基于TISitaraTMAM62x系列工业级处理器设计开发,采用ARMCortex-A53架构,主频最高可达1.4GHz;并集成了丰富的接口,可广泛应用于的工...
- 如何在 Linux 发行版中安装微信和 QQ?
-
很多人因为工作沟通的原因需要用到微信和QQ,那么如何在Linux发行版中安装微信和QQ呢?以下是一些尝试的解决方法。QQ上一个版本的QQLinux版还是在2009年,而在现在,基于N...
- MySQL:物理备份工具XBK(mysql 备份方案)
-
XBK的优缺点:XBK(PerconaXtraBackup)优点:1.免费2.热备:备份期间不阻塞innodb和XtraDB表,但会阻塞Myisam表3.物理备份:备份恢复快XBK缺点:1.不支持远...
- AMD锐龙9 9950X CPU AIDA64跑分曝光:比7950X最高快45%
-
IT之家6月26日消息,Anandtech论坛网友igor_kavinski本周一发布帖子,分享了AMD旗舰锐龙99950X处理器的AIDA64基准测试跑分,与当前基于Z...
- qemu linux内核(5.10.209)开发环境搭建
-
版本信息宿主机:ubuntu20.04.6LTS(FocalFossa)虚拟机:ubuntu20.04.6LTS(FocalFossa)安装宿主机的步骤省略,和一般的在vmware中安...
- iPhone 7成刷机神器,成功运行乌班图、Linux、安卓
-
在智能机刚开始流行的时候,很多手机发烧友都喜欢刷机,当时民间大神们制作了特别多优化的ROM。后来随着手机硬件的逐步提升,以及厂商们对系统的大力优化,让大家对于刷机的兴趣也越来越少。不知道大家还记得这部...
- 12 款最佳免费开源 Linux 渲染器 | 火狐浏览器 130.0 版本更新
-
12款最佳免费开源Linux渲染器Linux的一大优势在于其拥有丰富的开源软件,可以满足艺术家、摄影师、动画师和设计师的需求。凭借价格低廉的硬件、免费的软件以及少量的才能和灵感,任何人都可以创...
- Linux中xargs 命令详解与实用场景
-
xargs是Linux系统中常用的命令行工具之一,它能够从标准输入构造参数列表并传递给其他命令使用,是处理批量数据操作时的重要利器。一、xargs的基本语法xargs[OPTION]...[C...
- Linux 磁盘扩容(非LVM)方式(linux扩容lvm磁盘容量)
-
今天接到一个客户的需求,CentOS的/分区容量太小了,OA系统所有的数据都在这下面,由于当时前同事给客户安装系统时采用了标准分区,而不是LVM逻辑卷,所以不支持在线扩容。df-hT查看磁盘使...
- 一周热门
- 最近发表
-
- ARM64内核内存布局图(ARM64内核内存布局图解)
- ARM64 linux 调试串口通信(ARM64 linux 调试串口通信实验报告)
- Gentoo Linux 终止对 Itanium IA-64 体系的支持
- 如何检查 Linux 系统是 32 位还是 64 位?这9个命令查的又快又准!
- 调出好画面!带你玩转飞凌嵌入式AM62x开发板的显示接口
- 带你玩转AM62x开发板的显示接口——LVDS的显示和修改方式
- AM335x继任者?AM6254性能解析(am2361p)
- 如何在 Linux 发行版中安装微信和 QQ?
- MySQL:物理备份工具XBK(mysql 备份方案)
- AMD锐龙9 9950X CPU AIDA64跑分曝光:比7950X最高快45%
- 标签列表
-
- linux 远程 (37)
- u盘 linux (32)
- linux 登录 (34)
- linux 路径 (33)
- linux 文件命令 (35)
- linux 是什么 (35)
- linux 界面 (34)
- 查看文件 linux (35)
- linux 语言 (33)
- linux代码 (32)
- linux 查看命令 (33)
- 关闭linux (34)
- root linux (33)
- 删除文件 linux (35)
- linux 主机 (34)
- linux与 (33)
- linux 函数 (35)
- linux .ssh (35)
- cpu linux (35)
- 查看linux 系统 (32)
- linux 防火墙 (33)
- linux 手机 (32)
- linux 镜像 (34)
- linux mac (32)
- linux ip地址 (34)