8 种 Java 内存溢出之八 -Kill process or sacrifice child

简介: 8 种 Java 内存溢出之八 -Kill process or sacrifice child

8.1 Kill process or sacrifice child 概述

为了理解这个报错, 我们需要复习一下操作系统基础知识. 正如你所知, 操作系统是建立在进程的概念之上的. 这些过程是由多个内核作业引导的,其中一个以内存杀手 (out of memory killer) 命名的 worker 在这个特殊的情况下是我们感兴趣的。

这个内核作业会在极低的内存条件下消灭您的进程. 当检测到这种情况时,内存杀手就会被激活,并选择一个进程来杀死。使用一组启发式算法对所有进程进行评分,并选择得分最差的一个作为目标。OutOfMemoryError: Kill process or sacrifice child这与我们的 OOM 手册中涉及的其他错误不同,因为它不是由 JVM 触发或代理的,而是构建在操作系统内核中的安全网络。

当可用的虚拟内存 (包括 swap) 被消耗到整个操作系统的稳定性受到威胁的程度时,就会产生 OutOfMemoryError: Kill process or sacrifice child 错误。在这种情况下,内存杀手会选择这个流氓进程并杀死它。

8.2 原因

默认情况下,Linux 内核允许进程请求比当前系统中可用的更多的内存。考虑到大多数进程实际上从未使用它们所分配的所有内存,因此这在现实世界是很有意义的。 最简单的例子就是宽带运营商。他们向所有用户提供了 100Mbit 的下载承诺,远远超过了网络中实际的带宽(超卖)。再次打赌,用户绝不会同时使用他们分配的下载极限值。因此,一个 10Gbit 链接可以成功地服务于多于我们简单的数学算出来的 100 个用户。

这种方法的副作用是可见的,比如某些程序耗尽了系统内存。这可能导致极低的内存状态,没有页 (page) 可以分配到进程。您可能遇到过这样的情况,甚至连根帐户也不能杀死这个讨厌的任务. 为了防止这种情况,这个杀手激活了,并确认把这个流氓进程杀死.

您可以从 这篇 RedHat 文档 中阅读更多关于微调“内存杀手”行为的信息。

现在我们有了背景知识,那么你怎么知道是什么触发了“杀手”,并在凌晨 5 点叫醒了你? 激活的一个常见触发器隐藏在操作系统配置中。当您在 /proc/sys/vm/overcommit_memory 中检查配置时,您有了第一个提示 —— 特定的这个值表明是否允许所有 malloc()调用成功。注意,在 proc 文件系统中参数的路径取决于受更改影响的系统。超限提交配置允许为这个流氓过程分配越来越多的内存,最终可能触发“内存杀手”来完成它想要做的事情。

8.3 示例

当您在 Linux 上编译并启动以下 Java 代码片段时(我使用了最新的稳定 Ubuntu 版本):

package eu.plumbr.demo;
 
public class OOM {
 
public static void main(String[] args) {
    java.util.List<int[]> l = new java.util.ArrayList();
    for (int i = 10000; i<100000; i++) {
        try {
            l.add(new int[100_000_000]);
        } catch (Throwable t) {
            t.printStackTrace();
        }
    }
}
}
JAVA

然后您将在系统日志中看到类似下列的错误 (/var/log/kern.login) 的例子:

Jun  4 07:41:59 plumbr kernel: [70667120.897649] Out of memory: Kill
process 29957 (java) score 366 or sacrifice child
Jun  4 07:41:59 plumbr kernel: [70667120.897701] Killed process
29957 (java) total-vm:2532680kB, anon-rss:1416508kB, file-rss:0kB
MIPSASM

注意,您可能需要调整 swapfile 和堆大小,在我们的测试用例中,我们使用了一个由 -Xmx2g 指定的 2g 堆,并有如下的 swap 配置:

swapoff -a
dd if=/dev/zero of=swapfile bs=1024 count=655360
mkswap swapfile
swapon swapfile
SHELL

8.4 解决方案

有几种方式来解决这类场景. 解决这个问题的第一个也是最直接的方法是将系统迁移到具有更多内存的实例上。

其他的可能性包括 对 OOM 杀手进行微调,在几个小的实例上横向扩展负载,或者减少应用程序的内存需求。

我们不愿意推荐的一个解决方案是增加交换空间。当您回想起 Java 是一种垃圾收集的语言时,这个解决方案似乎就不那么有利可图了. 现代 GC 算法在物理内存中运行效率很高,但是当处理 swap 分配时,效率会受到重创。Swapping 会增加 GC 暂停的长度高达几个数量级,所以在选择到这个解决方案之前,您应该三思而后行。

相关文章
|
6天前
|
安全 Java 程序员
深入理解Java内存模型与并发编程####
本文旨在探讨Java内存模型(JMM)的复杂性及其对并发编程的影响,不同于传统的摘要形式,本文将以一个实际案例为引子,逐步揭示JMM的核心概念,包括原子性、可见性、有序性,以及这些特性在多线程环境下的具体表现。通过对比分析不同并发工具类的应用,如synchronized、volatile关键字、Lock接口及其实现等,本文将展示如何在实践中有效利用JMM来设计高效且安全的并发程序。最后,还将简要介绍Java 8及更高版本中引入的新特性,如StampedLock,以及它们如何进一步优化多线程编程模型。 ####
14 0
|
1月前
|
存储 Java 编译器
Java内存模型(JMM)深度解析####
本文深入探讨了Java内存模型(JMM)的工作原理,旨在帮助开发者理解多线程环境下并发编程的挑战与解决方案。通过剖析JVM如何管理线程间的数据可见性、原子性和有序性问题,本文将揭示synchronized关键字背后的机制,并介绍volatile关键字和final关键字在保证变量同步与不可变性方面的作用。同时,文章还将讨论现代Java并发工具类如java.util.concurrent包中的核心组件,以及它们如何简化高效并发程序的设计。无论你是初学者还是有经验的开发者,本文都将为你提供宝贵的见解,助你在Java并发编程领域更进一步。 ####
|
2月前
|
缓存 easyexcel Java
Java EasyExcel 导出报内存溢出如何解决
大家好,我是V哥。使用EasyExcel进行大数据量导出时容易导致内存溢出,特别是在导出百万级别的数据时。以下是V哥整理的解决该问题的一些常见方法,包括分批写入、设置合适的JVM内存、减少数据对象的复杂性、关闭自动列宽设置、使用Stream导出以及选择合适的数据导出工具。此外,还介绍了使用Apache POI的SXSSFWorkbook实现百万级别数据量的导出案例,帮助大家更好地应对大数据导出的挑战。欢迎一起讨论!
193 1
|
2月前
|
存储 监控 算法
Java中的内存管理:理解Garbage Collection机制
本文将深入探讨Java编程语言中的内存管理,着重介绍垃圾回收(Garbage Collection, GC)机制。通过阐述GC的工作原理、常见算法及其在Java中的应用,帮助读者提高程序的性能和稳定性。我们将从基本原理出发,逐步深入到调优实践,为开发者提供一套系统的理解和优化Java应用中内存管理的方法。
|
8天前
|
存储 监控 算法
Java内存管理深度剖析:从垃圾收集到内存泄漏的全面指南####
本文深入探讨了Java虚拟机(JVM)中的内存管理机制,特别是垃圾收集(GC)的工作原理及其调优策略。不同于传统的摘要概述,本文将通过实际案例分析,揭示内存泄漏的根源与预防措施,为开发者提供实战中的优化建议,旨在帮助读者构建高效、稳定的Java应用。 ####
22 8
|
14天前
|
Java
java内存区域
1)栈内存:保存所有的对象名称 2)堆内存:保存每个对象的具体属性 3)全局数据区:保存static类型的属性 4)全局代码区:保存所有的方法定义
20 1
|
28天前
|
缓存 算法 Java
本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制
在现代软件开发中,性能优化至关重要。本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制。通过调整垃圾回收器参数、优化堆大小与布局、使用对象池和缓存技术,开发者可显著提升应用性能和稳定性。
45 6
|
5天前
|
存储 监控 算法
Java内存管理的艺术:深入理解垃圾回收机制####
本文将引领读者探索Java虚拟机(JVM)中垃圾回收的奥秘,解析其背后的算法原理,通过实例揭示调优策略,旨在提升Java开发者对内存管理能力的认知,优化应用程序性能。 ####
20 0
|
1月前
|
存储 缓存 安全
Java内存模型(JMM):深入理解并发编程的基石####
【10月更文挑战第29天】 本文作为一篇技术性文章,旨在深入探讨Java内存模型(JMM)的核心概念、工作原理及其在并发编程中的应用。我们将从JMM的基本定义出发,逐步剖析其如何通过happens-before原则、volatile关键字、synchronized关键字等机制,解决多线程环境下的数据可见性、原子性和有序性问题。不同于常规摘要的简述方式,本摘要将直接概述文章的核心内容,为读者提供一个清晰的学习路径。 ####
40 2
|
1月前
|
存储 安全 Java
什么是 Java 的内存模型?
Java内存模型(Java Memory Model, JMM)是Java虚拟机(JVM)规范的一部分,它定义了一套规则,用于指导Java程序中变量的访问和内存交互方式。
68 1