`

【赵强老师】如何分析Java的内存溢出问题

阅读更多

一、什么是内存溢出?

内存溢出(OOM:out of memory)通俗理解就是内存不够,通常在运行大型软件或游戏时,软件或游戏所需要的内存远远超出了你主机内安装的内存所承受大小,就叫内存溢出。

在Java中,将会产生java.lang.OutOfMemoryError。看下关于的官方说明: Thrown when the Java Virtual Machine cannot allocate an object because it is out of memory, and no more memory could be made available by the garbage collector. 意思就是说,当JVM因为没有足够的内存来为对象分配空间并且垃圾回收器也已经没有空间可回收时,就会抛出这个error(注:非exception,因为这个问题已经严重到不足以被应用处理)。

二、为什么产生OOM?

为什么会没有内存了呢?原因不外乎有两点:

  • 分配的少了:比如虚拟机本身可使用的内存(一般通过启动时的VM参数指定)太少。
  • 应用用的太多,并且用完没释放,浪费了。此时就会造成内存泄露或者内存溢出。

在Java语言中,由于存在了垃圾自动回收机制,所以,我们一般不用去主动释放不用的对象所占的内存,也就是理论上来说,是不会存在“内存泄露”的。但是,如果编码不当,比如,将某个对象的引用放到了全局的Map中,虽然方法结束了,但是由于垃圾回收器会根据对象的引用情况来回收内存,导致该对象不能被及时的回收。如果该种情况出现次数多了,就会导致内存溢出,比如系统中经常使用的缓存机制。Java中的内存泄露,不同于C++中的忘了delete,往往是逻辑上的原因泄露。

三、如何分析Java OOM?

在故障定位(尤其是out of memory)和性能分析的时候,经常会用到一些文件来帮助我们排除代码问题。这些文件记录了JVM运行期间的内存占用、线程执行等情况,这就是我们常说的dump文件。常用的有heap dump和thread dump(也叫javacore,或java dump)。我们可以这么理解:heap dump记录内存信息的,thread dump是记录CPU信息的。这里我们重点介绍heap dump。
heap dump文件是一个二进制文件,它保存了某一时刻JVM堆中对象使用情况。HeapDump文件是指定时刻的Java堆栈的快照,是一种镜像文件。Heap Analyzer工具通过分析HeapDump文件,哪些对象占用了太多的堆栈空间,来发现导致内存泄露或者可能引起内存泄露的对象。

四、案例

首先,我们来开发一段Java程序。

import java.util.*;

public class Test {

 public static void main(String[] args) {
  List<String> list = new ArrayList<String>();
  
  int i = 0;
  while (true) {
   list.add(new String("test"));  }
 }

}

 

 

使用下面的命令运行该程序时设置JVM的堆内存(heap size)的极限值为10M(-Xmx10m)。

java -Xmx10m Test

 

很快,程序将会产生OOM的错误,如下图所示:

五、如何生成Head Dump文件?

我们可以在运行Java程序的时候,加入下面的参数:

-XX:+HeapDumpOnOutOfMemoryError

 

此参数是帮助生成dump文件,程序启动后直到抛出OOM异常。异常抛出后,在程序的classpath下会生成以一个以.hprof结尾的文件,如:java_pid4504.hprof,这就是我们需要的dump文件。

如下图所示:

六、使用IBM heapAnalyzer分析Head Dump文件

IBM heapAnalyzer(https://www.ibm.com/support/pages/ibm-heapanalyzer)是IBM开发的强大的内存dump分析工具,,IBM heapAnalyzer是通过分析OOM后的Java heap dump文件的,通过对dump文件的分析找到内存可能泄露的点。

启动IBM heapAnalyzer,并导入刚才生成的Heap Dump文件,如下图所示。

通过分析我们会发现,系统94.19%的内存都被一个ArrayList占用了(里面保存的都是Object)。这里就有可能是一个内存的溢出点。当然,我们这个例子非常典型,在实际工作可能没有这么明显,需要具体问题具体分析。

分享到:
评论

相关推荐

    传智播客 赵强 Oracle课件

    10. **实战项目**:通过赵强老师的课件,你将有机会参与到实际的项目中,将所学知识应用于解决实际问题,提高解决复杂数据库问题的能力。 赵强老师的Oracle课件全面覆盖了Oracle数据库的学习路径,无论你是初学者...

    赵强老师的Oracle课件

    想要好好地学习Oracle数据库的朋友呀,你错过了她就太不值得了。里面有好多的Oracle操作命令可能你都没接触过吧。好了,话不多多说。坚信资料不错!你,值得拥有!OK.还有,之所有要你2分打赏,是我给了你这么好的...

    精通JSP编程 作者赵强 编

    精通JSP编程 作者赵强 编 12-18节

    [1120][赵强精通JSP编程][37M]

    根据提供的文件信息,我们可以推断出这是一份与Java Server Pages (JSP)相关的学习资料介绍,特别是关于赵强编写的《精通JSP编程》这本书的相关信息。下面将基于这个理解来生成相关知识点。 ### 一、JSP基础概念 ...

    精通JSP编程 作者赵强

    《精通JSP编程》是赵强先生的一部深入解析JSP技术的专业著作,该书针对JSP编程进行了全面且深入的讲解,旨在帮助读者掌握JSP的核心概念和技术,提升Web应用开发能力。根据提供的文件名列表,我们可以推测书籍的章节...

    java代码-46 赖赵强

    【标题】"java代码-46 赖赵强"所指的可能是一个关于Java编程的项目或示例,由开发者赖赵强创建。在这个项目中,他可能分享了一段特定的Java代码,用于解决某种问题或者实现一个功能。这个标题暗示了这是一个与Java...

    Oracle学习笔记(传智播客 赵强)

    而`IN`则不会有问题,即使子查询结果中有NULL。如果要排除NULL,可以在子查询中添加`WHERE`子句。 这些优化技巧能帮助提升SQL查询的效率,减少数据库资源的消耗,尤其对于大型Oracle数据库来说,这些最佳实践是必不...

    java代码-46赖赵强 例3-4

    在本压缩包中,我们关注的是“java代码-46赖赵强 例3-4”的主题,这可能是一个编程教学实例或者一个特定问题的解决方案。"赖赵强"可能是这个例子的创建者或者讲解者,而“例3-4”则表明这是一个系列教程或示例集中的...

    java代码-46 赖赵强 1-100里面的素数

    【标题】:“java代码-46 赖赵强 1-100里面的素数” 这个项目是关于使用Java编程语言找出1到100之间的所有素数。素数是指除了1和它本身以外没有其他正因数的自然数。在数学中,素数具有重要的理论价值和实际应用,...

    java 的基础用运

    【Java的基础运用】 Java是一种广泛使用的面向对象的编程语言,尤其在Web应用程序开发领域有着重要地位。在《JSP程序设计》这门课程中,学生将深入学习如何使用Java Server Pages (JSP) 技术来构建动态网站和Web应用...

    java代码-46 赖赵强 求100以内所有偶数的和

    在本项目中,标题"java代码-46 赖赵强 求100以内所有偶数的和"表明这是一个使用Java编程语言编写的程序,由名为“赖赵强”的开发者创建,目的是计算100以内所有偶数的总和。这个任务是一个基础的算法实现,适用于...

    Oracle数据库赵强视频教程【3天】

    教程名称:Oracle 数据库赵强视频教程【3天】教程目录:【】Oracle安装与管理、SQL语句(赵强)【】Orcale存储过程jdbc与Orcale大文本操作等(赵强)【】SQL简单查询触发器视图(赵强)  资源太大,传百度网盘了,链接在...

    java代码-46赖赵强将8进制数转换为十进制数

    虽然Java的内置函数已经优化过,但自定义实现可能需要考虑优化,如使用长整型处理大数,避免溢出问题。 10. **扩展应用**: - 这种转换方法不仅限于八进制到十进制,也可以扩展到其他任意基数到十进制的转换,只需...

Global site tag (gtag.js) - Google Analytics