Java中的内存泄漏

Java.Lang.OutOfMemoryError: Java Heap Space

Java应用程序只允许使用有限的内存。此限制在应用程序启动期间指定。为了使事情更复杂,Java内存被分成两个不同的区域。这些区域称为永久生成区域(permgene和Permgen):

这些区域的大小是在Java虚拟机(JVM)启动期间设置的,可以通过指定JVM参数-Xmx和-XX:MaxPermSize进行定制。如果未显式设置大小,则将使用特定于平台的默认值。

这个java.lang.OutOfMemoryError:当应用程序尝试向堆空间区域添加更多数据,但空间不足时,将触发Java堆空间错误。

请注意,可能有大量可用的物理内存,但是java.lang.OutOfMemoryError:每当JVM达到堆大小限制时,就会抛出Java堆空间错误。

是什么引起内存泄露的?

最常见的原因是java.lang.OutOfMemoryError:Java heap space error –您尝试将XXL应用程序放入S大小的Java堆空间中。只是Java应用程序需要更多的空间来操作。此OutOfMemoryError消息的其他原因更为复杂,是由编程错误引起的:

  • 使用量/数据量激增。该应用程序设计为处理一定数量的用户或一定数量的数据。当用户数量或数据量突然达到峰值并超过预期阈值时,在峰值停止运行并触发java.lang.OutOfMemoryError:Java堆空间错误。
  • 内存泄漏。特定类型的编程错误将导致应用程序不断消耗更多内存。每次使用应用程序的泄漏功能时,它都会将一些对象留在Java堆空间中。随着时间的推移,泄漏的对象会消耗所有可用的Java堆空间,并触发已经熟悉的java.lang.OutOfMemoryError:Java堆空间错误。

内存泄漏代码示例

第一个例子非常简单–下面的Java代码尝试分配一个2M整数数组。当您编译它并使用12MB的Java堆空间(Java-Xmx12m-OOM)启动时,它将失败java.lang.OutOfMemoryError:Java堆空间消息。有了13MB的Java堆空间,程序运行得很好。

class OOM {
 static final int SIZE=2*1024*1024;
 public static void main(String[] a) {
  int[] i = new int[SIZE];
  }
}

第二个也是更现实的例子是内存泄漏在Java中,当开发人员创建和使用新的对象(如new Integer(5))时,他们不必自己分配内存—这是由Java虚拟机(JVM)负责的。在应用程序的生命周期中,JVM会定期检查内存中哪些对象仍在使用,哪些对象没有使用。未使用的对象可以丢弃,内存可以回收并再次使用。这个过程称为垃圾回收。JVM中负责收集的相应模块称为垃圾收集器(GC)。

Java的自动内存管理依赖于GC定期查找未使用的对象并将其删除。简单地说,java 内存泄露是指应用程序不再使用某些对象,但垃圾回收无法识别它的情况。因此,这些未使用的对象将无限期地保留在Java堆空间中。这起连环碰撞最终会触发java.lang.OutOfMemoryError:Java堆空间错误。

构建一个满足内存泄漏定义的Java程序相当容易:

class KeylessEntry {

  static class Key {
   Integer id;

   Key(Integer id) {
     this.id = id;
   }

   @Override
   public int hashCode() {
     return id.hashCode();
   }
  }

  public static void main(String[] args) {
   Map m = new HashMap();
   while (true)
     for (int i = 0; i < 10000; i++)
      if (!m.containsKey(new Key(i)))
        m.put(new Key(i), "Number:" + i);
  }
}

当您执行上面的代码时,您可能希望它永远运行而不会出现任何问题,假设天真的缓存解决方案只将底层映射扩展到10000个元素,除此之外,所有的键都已经存在于HashMap中。但是,实际上,由于Key类在hashCode()旁边没有适当的equals()实现,所以元素将继续被添加。

结果,随着时间的推移,随着泄漏代码的不断使用,“缓存”结果最终会消耗大量Java堆空间。当泄漏的内存填满堆区域中的所有可用内存,而垃圾回收无法清理它时java.lang.OutOfMemoryError:引发Java堆空间。

解决方案很简单–添加与下面类似的equals()方法的实现,您就可以开始了。但在你找到病因之前,你肯定会失去一些珍贵的脑细胞。

@Override
public boolean equals(Object o) {
  boolean response = false;
  if (o instanceof Key) {
   response = (((Key)o).id).equals(this.id);
  }
  return response;
}

内存溢出怎么解决?

在某些情况下,分配给JVM的堆的数量不足以满足在JVM上运行的应用程序的需要。在这种情况下,您应该只分配更多的堆—请参阅本章末尾的部分了解如何实现这一点。

然而,在许多情况下,提供更多的Java堆空间并不能解决问题。例如,如果应用程序包含内存泄漏,则添加更多堆只会推迟java.lang.OutOfMemoryError:Java堆空间错误。此外,增加Java堆空间量也会增加GC暂停的长度,从而影响应用程序的吞吐量或延迟。

如果您希望解决Java堆空间的底层问题,而不是掩盖症状,那么您需要找出代码的哪一部分负责分配最多的内存。换句话说,你需要回答以下问题:

  1. 哪些对象占据堆的大部分
  2. 在源代码中分配这些对象

在这一点上,一定要在你的日历中清除几天(或者-在项目符号列表下面自动查看)。下面是一个粗略的流程大纲,可以帮助您回答上述问题:

  • 获得安全许可,以便从JVM执行堆转储。“dump转储”基本上是堆内容的快照,您可以对其进行分析。因此,这些快照可能包含机密信息,如密码、信用卡号码等,因此出于安全原因,获取此类转储甚至可能不可能。
  • 在适当的时候把垃圾处理掉。准备好获取一些转储,因为当在错误的时间执行时,堆转储包含大量的噪声,实际上可能是无用的。另一方面,每个堆转储都会完全“freezes冻结”JVM,所以不要占用太多,否则最终用户将面临性能问题。
  • 找一台能装垃圾的机器。当您的JVM使用例如8GB的堆时,您需要一台大于8GB的机器来分析堆内容。启动转储分析软件(我们推荐Eclipse MAT,但也有同样好的替代品)。
  • 检测堆的最大使用者的GC根路径。我们在这里的另一篇文章中讨论了这一活动。这对初学者来说尤其困难,但实践将使你了解结构和导航机制。
  • 接下来,您需要弄清楚源代码中潜在危险的大量对象被分配到哪里。如果您对应用程序的源代码有很好的了解,那么您将能够在几次搜索中做到这一点。

或者,我们建议使用plumber,这是唯一一个具有自动根本原因检测功能的Java监控解决方案。在其他性能问题中,它包罗万象java.lang.OutOfMemoryErrors并自动为您提供有关最需要内存的数据结构的信息。

Plumber负责在后台收集必要的数据——这包括关于堆使用情况的相关数据(只有对象布局图,没有实际数据),还有一些甚至在堆转储中都找不到的数据。它还为您执行必要的数据处理—在运行中,只要JVM遇到java.lang.OutOfMemoryError. 这里有一个例子java.lang.OutOfMemoryError管道工事故警报:

无需任何其他工具或分析,您可以看到:

  • 哪些对象消耗的内存最多
  • 在哪里分配这些对象(它们中的大多数在MetricManagerImpl类中分配,第304行)
  • 当前引用这些对象的是什么(到GC根的完整引用链)

有了这些信息,您就可以放大潜在的根本原因,并确保将数据结构缩减到适合您的内存池的级别。

然而,当您从内存分析或阅读plumber报告得出的结论是内存使用是合法的,并且源代码中没有什么可更改的,那么您需要允许JVM有更多的Java堆空间来正常运行。在这种情况下,更改JVM启动配置并添加(或增加值,如果存在):

-Xmx1024m

上述配置将为应用程序提供1024MB的Java堆空间。可以使用g或g表示GB,m或m表示MB,k或k表示KB。例如,以下所有内容都相当于最大Java堆空间为1GB:

  java -Xmx1073741824 com.mycompany.MyClass
  java -Xmx1048576k com.mycompany.MyClass
  java -Xmx1024m com.mycompany.MyClass
  java -Xmx1g com.mycompany.MyClass

以上就是Java中的内存泄漏的详细内容,更多关于Java 内存泄漏的资料请关注我们其它相关文章!

(0)

相关推荐

  • Android性能优化之利用Rxlifecycle解决RxJava内存泄漏详解

    前言: 其实RxJava引起的内存泄漏是我无意中发现了,本来是想了解Retrofit与RxJava相结合中是如何通过适配器模式解决的,结果却发现了RxJava是会引起内存泄漏的,所有想着查找一下资料学习一下如何解决RxJava引起的内存泄漏,就查到了利用Rxlifecycle开源框架可以解决,今天周末就来学习一下如何使用Rxlifecycle. 引用泄漏的背景: RxJava作为一种响应式编程框架,是目前编程界网红,可谓是家喻户晓,其简洁的编码风格.易用易读的链式方法调用.强大的异步支持等使得R

  • 解析Java的JNI编程中的对象引用与内存泄漏问题

    JNI,Java Native Interface,是 native code 的编程接口.JNI 使 Java 代码程序可以与 native code 交互--在 Java 程序中调用 native code:在 native code 中嵌入 Java 虚拟机调用 Java 的代码. JNI 编程在软件开发中运用广泛,其优势可以归结为以下几点: 利用 native code 的平台相关性,在平台相关的编程中彰显优势. 对 native code 的代码重用. native code 底层操作

  • Java DWR内存泄漏问题解决方案

    机器跑了一晚上,发现有崩溃现象,由于页面内有动态绘图功能,我怀疑是绘图原因,但是今天上午有人提醒我才想到,是不是间隔调用时DWR产生了内存泄漏问题? 网上查了一下貌似大家都在讨论这个问题,之前我也挺老手说过DWR有内存问题,可是没有遇到过.原来DWR在间隔调用这种情况下会有问题! 按照大家的说法,修改engine.js配置文件来解决问题,目前我也修改了一下,修改方法如下: 在 dwr.engine._sendData = function(batch) {} 内有,这样几行代码: // Get

  • java虚拟机内存溢出及泄漏实例

    测试参数设置: 1.循环调用new A()实现堆溢出,java.lang.OutOfMemoryError: Java heap space, 虚拟机参数:-Xms1M -Xmx1M -XX:+HeapDumpOnOutOfMemoryError,解释:将-Xmx和-Xms设置为一样可以避免堆自动扩展,-XX:+HeapDumpOnOutOfMemoryError可以让虚拟机在出现内存溢出异常时Dump出当前的堆内存转储快照 // while (true){ // new A().do2();

  • 深入理解Java垃圾回收机制以及内存泄漏

    前言 在segmentfault上看到一个问题:java有完善的GC机制,那么在java中是否会出现内存泄漏的问题,以及能否给出一个内存泄漏的案例.本问题视图给出此问题的完整答案. 垃圾回收机制简介 在程序运行过程中,每创建一个对象都会被分配一定的内存用以存储对象数据.如果只是不停的分配内存,那么程序迟早面临内存不足的问题.所以在任何语言中,都会有一个内存回收机制来释放过期对象的内存,以保证内存能够被重复利用. 内存回收机制按照实现角色的不同可以分为两种,一种是程序员手动实现内存的释放(比如C语

  • 浅析Java中的内存泄漏

    ava最明显的一个优势就是它的内存管理机制.你只需简单创建对象,java的垃圾回收机制负责分配和释放内存.然而情况并不像想像的那么简单,因为在Java应用中经常发生内存泄漏. 本教程演示了什么是内存泄漏,为什么会发生内存泄漏以及如何预防内存泄漏. 什么是内存泄漏? 定义:如果对象在应用中不再被使用,但由于它们在其他地方被引用,垃圾回收却不能移除它们(这样就造成了很多内存不能释放,从而导致内存溢出的现象.译注). 要理解这一定义,我们需要理解内存中对象的状态.下图说明了那些是未使用,那些是未引用.

  • Java中关于内存泄漏出现的原因汇总及如何避免内存泄漏(超详细版)

    Android 内存泄漏总结 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题.内存泄漏大家都不陌生了,简单粗俗的讲,就是该被释放的对象没有释放,一直被某个或某些实例所持有却不再被使用导致 GC 不能回收.最近自己阅读了大量相关的文档资料,打算做个 总结 沉淀下来跟大家一起分享和学习,也给自己一个警示,以后 coding 时怎么避免这些情况,提高应用的体验和质量. 我会从 java 内存泄漏的基础知识开始,并通过具体例子来说明 Android 引起内存泄漏的各种原因,以

  • Java中由substring方法引发的内存泄漏详解

    内存溢出(out of memory ) :通俗的说就是内存不够用了,比如在一个无限循环中不断创建一个大的对象,很快就会引发内存溢出. 内存泄漏(leak of memory) :是指为一个对象分配内存之后,在对象已经不在使用时未及时的释放,导致一直占据内存单元,使实际可用内存减少,就好像内存泄漏了一样. 由substring方法引发的内存泄漏 substring(int beginIndex, int endndex )是String类的一个方法,但是这个方法在JDK6和JDK7中的实现是完全

  • 排查Java应用内存泄漏问题的步骤

    什么是内存泄漏 内存泄漏是指java应用的堆内存使用率持续升高,直至内存溢出. 内存泄漏的的原因可能有多种 分配给应用程序的内存本身过小.而应用的业务代码,确实需要生成大量的对象 代码bug,某些需要被回收的对象,由于代码bug,却持续的被引用,导致java虚拟机无法回收这些对象.从而撑爆内存 无论哪种内存泄露,我们的解决方法都是要定位到具体是什么对象,占用了大量内存,从而方便我们基于此进行代码分析,debug,找出代码问题. 而能够帮助我们实现这一目的的方式就是获取java应用的内存 dump

  • Java内存泄漏问题处理方法经验总结

    JVM问题,一般会有三种情况,目前遇到了两种,线程溢出和JVM不够用 1.线程溢出:unable to create new native thread 1.1问题描述: 系统在1月4号左右,突然发现会产生内存溢出问题,从日志上看,错误信息为: 导致系统不能使用,对外不能相应,但是观察gc等又处于正常情况,free 系统内存也正常.开始重启机器进行解决,真正的原因查找,过程比较坎坷,经历也比较痛苦. 1.2 问题解决 pstree查看线程数,发现系统线程数不断增长,直到OOM. 命令:pstre

随机推荐