8 种 Java- 内存溢出之四 -Metaspace

简介: 8 种 Java- 内存溢出之四 -Metaspace

4.1 java.lang.OutOfMemoryError: Metaspace 概述

Java 应用只允许使用有限的内存. 你的应用可以用的准确的内存大小在启动的时候指定. 展开来说, Java 内存被分成不同的区域, 具体如下图:

所有的这些区域, 包括元空间 (metaspace) 区域, 可以在 JVM 启动的时候指定. 如果你没有指定这些的大小, 平台相关的默认配置会被应用.

java.lang.OutOfMemoryError: Metaspace 消息表明 Metaspace 区内存耗尽.

4.2 原因

如果你不是 Java 领域的新手, 你可能会熟悉 Java 内存管理的另一个叫做: PermGen 的概念. 从 Java 8 开始, Java 的内存模型发生明显改变. 引入一个叫做 Metaspace 的新的内存区域, Permgen 被移除. 这个变更是基于多种原因的, 包括但不限于:

  • PermGen 需要的内存大小难以预测. 它导致有可能由于内存不足触发 java.lang.OutOfMemoryError: Permgen 错误或预留过多导致浪费资源.
  • GC 性能的提升, 在没有 GC 暂停和元数据的特定迭代器的情况下启用并发类数据分配.
  • 支持进一步优化,如 G1 并发类卸载。

所以, 如果你熟悉 PermGen, 那么你需要知道的就是 – Java 8 之前版本在 PermGen 里存在的一切东西(组成类的名字和字段, 方法的字节码, 常量池信息, 对象数组和类型数组, 以及实时编译优化) – 现在都在 Metaspace 里.

如你所见, Metaspace 大小需求取决于加载的类的数量和这些类声明的大小. 所以, 很明显 java.lang.OutOfMemoryError: Metaspace 的主要原因是: 太多类, 或太大的类被加载到 Metaspace 中.

4.3 案例

就如我们在之前解释的那样, Metaspace 使用率与加载到 JVM 中的类的数量强相关. 下列代码就是最简单的例子:

public class Metaspace {
    static javassist.ClassPool cp = javassist.ClassPool.getDefault();
 
    public static void main(String[] args throws Exception {
        for (int i=0; ;i++) {
            Class c = cp.makeClass("eu.plumbr.demo.Generated" + i).toClass();
        }
    }
}
JAVA

在这个例子中, 源代码循环遍历一个循环并在运行时生成类. 所有这些生成的类定义在持续地消耗 Metaspace. javassist库对类生成的复杂性进行了处理.

代码会持续生成新的类, 并加载他们的定义到 Metaspace 中直到空间被完全占满, java.lang.OutOfMemoryError: Metaspace抛出. 当在 Mac OS X, Java 1.8.0_05 上使用 -XX:MaxMetaspaceSize=64m 大概加载 70,000 个类会死掉.

4.4 解决方案

第一个解决因为 Metaspace 内存溢出的方案很明显. 如果应用耗尽了 Metaspace 的内存, 你应该增加 Metaspace 的大小. 调整应用运行配置, 调整下列参数:

-XX:MaxMetaspaceSize=512m

上述配置案例告诉 JVM, Metaspace 允许在抛出 OutOfMemoryError 的错误之前增长到 512MB.

另一个解决方案乍一看更容易. 你可以通过删除这个参数移除 Metaspace 的限制. 但是需要注意的是, 你这么做, 可能会导致 swap 的大量消耗和 / 或导致本机物理内存分配失败.

通过使用上述建议的 “快速修复”, 你只会通过隐藏 java.lang.OutOfMemoryError: Metaspace掩盖症状, 而不是从根本上解决问题.

如果应用程序内存泄漏或是加载的不合理的东西到 Metaspace, 上述解决方案实际上不会改善任何事情, 它只会推迟问题。

相关文章
|
3月前
|
存储 缓存 安全
Java内存模型深度解析:从理论到实践####
【10月更文挑战第21天】 本文深入探讨了Java内存模型(JMM)的核心概念与底层机制,通过剖析其设计原理、内存可见性问题及其解决方案,结合具体代码示例,帮助读者构建对JMM的全面理解。不同于传统的摘要概述,我们将直接以故事化手法引入,让读者在轻松的情境中领略JMM的精髓。 ####
64 6
|
3月前
|
存储 Java 编译器
Java内存模型(JMM)深度解析####
本文深入探讨了Java内存模型(JMM)的工作原理,旨在帮助开发者理解多线程环境下并发编程的挑战与解决方案。通过剖析JVM如何管理线程间的数据可见性、原子性和有序性问题,本文将揭示synchronized关键字背后的机制,并介绍volatile关键字和final关键字在保证变量同步与不可变性方面的作用。同时,文章还将讨论现代Java并发工具类如java.util.concurrent包中的核心组件,以及它们如何简化高效并发程序的设计。无论你是初学者还是有经验的开发者,本文都将为你提供宝贵的见解,助你在Java并发编程领域更进一步。 ####
|
2月前
|
安全 Java 程序员
深入理解Java内存模型与并发编程####
本文旨在探讨Java内存模型(JMM)的复杂性及其对并发编程的影响,不同于传统的摘要形式,本文将以一个实际案例为引子,逐步揭示JMM的核心概念,包括原子性、可见性、有序性,以及这些特性在多线程环境下的具体表现。通过对比分析不同并发工具类的应用,如synchronized、volatile关键字、Lock接口及其实现等,本文将展示如何在实践中有效利用JMM来设计高效且安全的并发程序。最后,还将简要介绍Java 8及更高版本中引入的新特性,如StampedLock,以及它们如何进一步优化多线程编程模型。 ####
53 0
|
3月前
|
存储 算法 Java
Java内存管理深度剖析与优化策略####
本文深入探讨了Java虚拟机(JVM)的内存管理机制,重点分析了堆内存的分配策略、垃圾回收算法以及如何通过调优提升应用性能。通过案例驱动的方式,揭示了常见内存泄漏的根源与解决策略,旨在为开发者提供实用的内存管理技巧,确保应用程序既高效又稳定地运行。 ####
|
2天前
|
存储 IDE Java
java设置栈内存大小
在Java应用中合理设置栈内存大小是确保程序稳定性和性能的重要措施。通过JVM参数 `-Xss`,可以灵活调整栈内存大小,以适应不同的应用场景。本文介绍了设置栈内存大小的方法、应用场景和注意事项,希望能帮助开发者更好地管理Java应用的内存资源。
16 1
|
7天前
|
Java Shell 数据库
【YashanDB 知识库】kettle 同步大表提示 java 内存溢出
【问题分类】数据导入导出 【关键字】数据同步,kettle,数据迁移,java 内存溢出 【问题描述】kettle 同步大表提示 ERROR:could not create the java virtual machine! 【问题原因分析】java 内存溢出 【解决/规避方法】 ①增加 JVM 的堆内存大小。编辑 Spoon.bat,增加堆大小到 2GB,如: if "%PENTAHO_DI_JAVA_OPTIONS%"=="" set PENTAHO_DI_JAVA_OPTIONS="-Xms512m" "-Xmx512m" "-XX:MaxPermSize=256m" "-
|
2月前
|
存储 监控 算法
Java内存管理深度剖析:从垃圾收集到内存泄漏的全面指南####
本文深入探讨了Java虚拟机(JVM)中的内存管理机制,特别是垃圾收集(GC)的工作原理及其调优策略。不同于传统的摘要概述,本文将通过实际案例分析,揭示内存泄漏的根源与预防措施,为开发者提供实战中的优化建议,旨在帮助读者构建高效、稳定的Java应用。 ####
56 8
|
2月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
2月前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
102 5
|
2月前
|
存储 算法 Java
Java内存管理深度解析####
本文深入探讨了Java虚拟机(JVM)中的内存分配与垃圾回收机制,揭示了其高效管理内存的奥秘。文章首先概述了JVM内存模型,随后详细阐述了堆、栈、方法区等关键区域的作用及管理策略。在垃圾回收部分,重点介绍了标记-清除、复制算法、标记-整理等多种回收算法的工作原理及其适用场景,并通过实际案例分析了不同GC策略对应用性能的影响。对于开发者而言,理解这些原理有助于编写出更加高效、稳定的Java应用程序。 ####

热门文章

最新文章