阿里面试官:解释一下内存泄露?我大意了,没有闪

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
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
初识Visual Leak Detector   灵活自由是C/C++语言的一大特色,而这也为C/C++程序员出了一个难题。当程序越来越复杂时,内存的管理也会变得越加复杂,稍有不慎就会出现内存问题。内存泄漏是最常见的内存问题之一。内存泄漏如果不是很严重,在短时间内对程序不会有太大的影响,这也使得内存泄漏问题有很强的隐蔽性,不容易被发现。然而不管内存泄漏多么轻微,当程序长时间运行时,其破坏力是惊人的,从性能下降到内存耗尽,甚至会影响到其他程序的正常运行。另外内存问题的一个共同特点是,内存问题本身并不会有很明显的现象,当有异常现象出现时已时过境迁,其现场已非出现问题时的现场了,这给调试内存问题带来了很大的难度。   Visual Leak Detector是一款用于Visual C++的免费的内存泄露检测工具。相比较其它的内存泄露检测工具,它在检测到内存泄漏的同时,还具有如下特点:   1、 可以得到内存泄漏点的调用堆栈,如果可以的话,还可以得到其所在文件及行号;   2、 可以得到泄露内存的完整数据;   3、 可以设置内存泄露报告的级别;   4、 它是一个已经打包的lib,使用时无须编译它的源代码。而对于使用者自己的代码,也只需要做很小的改动;   5、 他的源代码使用GNU许可发布,并有详尽的文档及注释。对于想深入了解堆内存管理的读者,是一个不错的选择。   可见,从使用角度来讲,Visual Leak Detector简单易用,对于使用者自己的代码,唯一的修改是#include Visual Leak Detector的头文件后正常运行自己的程序,就可以发现内存问题。从研究的角度来讲,如果深入Visual Leak Detector源代码,可以学习到堆内存分配与释放的原理、内存泄漏检测的原理及内存操作的常用技巧等。   本文首先将介绍Visual Leak Detector的使用方法与步骤,然后再和读者一起初步的研究Visual Leak Detector的源代码,去了解Visual Leak Detector的工作原理。   使用Visual Leak Detector(1.0)   下面让我们来介绍如何使用这个小巧的工具。   首先从网站上下载zip包,解压之后得到vld.h, vldapi.h, vld.lib, vldmt.lib, vldmtdll.lib, dbghelp.dll等文件。将.h文件拷贝到Visual C++的默认include目录下,将.lib文件拷贝到Visual C++的默认lib目录下,便安装完成了。因为版本问题,如果使用windows 2000或者以前的版本,需要将dbghelp.dll拷贝到你的程序的运行目录下,或其他可以引用到的目录。   接下来需要将其加入到自己的代码中。方法很简单,只要在包含入口函数的.cpp文件中包含vld.h就可以。如果这个cpp文件包含了stdafx.h,则将包含vld.h的语句放在stdafx.h的包含语句之后,否则放在最前面。如下是一个示例程序:   #include   void main()   {   …   }   接下来让我们来演示如何使用Visual Leak Detector检测内存泄漏。下面是一个简单的程序,用new分配了一个int大小的堆内存,并没有释放。其申请的内存地址用printf输出到屏幕上。   #include   #include   #include   void f()   {   int *p = new int(0x12345678);   printf("p=%08x, ", p);   }   void main()   {   f();   }   编译运行后,在标准输出窗口得到:   p=003a89c0   在Visual C++的Output窗口得到:   WARNING: Visual Leak Detector detected memory leaks!   ---------- Block 57 at 0x003A89C0: 4 bytes ---------- --57号块0x003A89C0地址泄漏了4个字节   Call Stack: --下面是调用堆栈   d:\test\testvldconsole\testvldconsole\main.cpp (7): f --表示在main.cpp第7行的f()函数   d:\test\testvldconsole\testvldconsole\main.cpp (14): main –双击以引导至对应代码处   f:\rtm\vctools\crt_bld\self_x8
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值