前言

如今不管是在面试还是在我们的工作中,OOM总是不断的出现在我们的视野中,所以我们有必要去了解一下导致OOM的原因以及一些基本的调整方法,大家可以通过下面的事例来了解一下什么样的代码会导致OOM,帮助我们以后在工作中能够通过异常信息来判断是JVM里面哪个区域出现了问题。

先介绍一下笔者的相关编码环境。

jdk:java version "1.8.0_121"

ide:IntelliJ IDEA 2019.1 (Community Edition)


正文

1.Java堆溢出

Java中的堆存储的都是对象实例,当我们不断的创建对象,而GC的时候又不能回收,当存储的对象大小超过了-Xmx的值,这时候则会出现OutOfMemoryError.[-XX:+HeapDumpOnOutOfMemoryError]参数可以让jvm出现内存溢出的时候dump出内存堆转储快照。

  1. /**
  2. * VM Args: -Xms10m -Xmx10m -XX:+HeapDumpOnOutOfMemoryError
  3. * @author wangzenghuang
  4. */
  5. public class HeapOOMDemo {
  6. public static void main(String[] args) {
  7. List<String> stringList = new ArrayList<>();
  8. while(true){
  9. stringList.add("str");
  10. }
  11. }
  12. }

运行结果,发生OOM,并且在我们项目的根目录dump出当前的内存堆快照

  1. java.lang.OutOfMemoryError: Java heap space
  2. Dumping heap to java_pid1376.hprof ...
  3. Heap dump file created [ bytes in 0.047 secs]
  4. Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
  5. at java.util.Arrays.copyOf(Arrays.java:)
  6. at java.util.Arrays.copyOf(Arrays.java:)
  7. at java.util.ArrayList.grow(ArrayList.java:)
  8. at java.util.ArrayList.ensureExplicitCapacity(ArrayList.java:)
  9. at java.util.ArrayList.ensureCapacityInternal(ArrayList.java:)
  10. at java.util.ArrayList.add(ArrayList.java:)
  11. at HeapOOMDemo.main(HeapOOMDemo.java:)
  12.  
  13. Process finished with exit code

简单解决思路

那么发生这个问题以后我们的解决思路有哪些呢?我们可以利用一些工具(例如Eclipse Memory Analyzer )来分析dump出的文件,一般来说,当生产环境发生OOM,比较常见的一个原因是发生了内存泄漏,用工具可以分析出泄露的对象到GC Root的引用链,从而定位到问题代码。假如经过分析后发现内存中的对象都是“必须存活”的对象,这时候就要思考下项目中是否把“-Xms跟-Xmx”设置得太小了(当然这里也不是随意调大,需要结合机器的物理内存情况),再者需要留意代码中是否有一些长生命周期的对象,从代码中优化内存消耗。

2.方法区溢出

在jvm的方法区中,它主要存放了类的信息,常量,静态变量等。在jdk8以前是通过“-XX:PermSize,-XX:MaxPermSize”来调整这个区域的值,但是从8开始呢,永久代的概念被MetaSpace(元空间)代替了,对应的参数也变成了“-XX:MetaspaceSize,-XX:MaxMetaspaceSize”。在这个例子中使用CGLib来动态生成一些类,方便我们实验操作。

  1. /**
  2. * VM Args: -XX:MetaspaceSize=5m -XX:MaxMetaspaceSize=5m
  3. * @author wangzenghuang
  4. */
  5. public class MethodAreaOOMDemo {
  6. public static void main(String[] args) {
  7. while(true){
  8. Enhancer enhancer = new Enhancer();
  9. enhancer.setSuperclass(OOMObject.class);
  10. enhancer.setUseCache(false);
  11. enhancer.setCallback(new MethodInterceptor() {
  12. public Object intercept(Object obj, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
  13. return methodProxy.invokeSuper(obj,objects);
  14. }
  15. });
  16. enhancer.create();
  17. }
  18. }
  19. static class OOMObject{}
  20. }

运行结果

  1. Exception in thread "main" java.lang.OutOfMemoryError: Metaspace

简单解决方法

这个问题的话,一般来说根据情况调整方法区的大小就行了,网上也有人说可以去掉MetaSpace的的大小限制,但是不建议这么干,毕竟不可控的事情我们要少点干,很容易给自己埋雷。

3.栈溢出

对于我们来说,还有一个熟悉的错误,那就是“StackOverflowError”,它是由线程请求的栈深度超过了jvm允许的最大范围而产生的。“-Xss”参数可以设置栈容量。

  1. /**
  2. * VM Args: -Xss128k
  3. * @author wangzenghuang
  4. */
  5. public class StackOFDemo {
  6. ;
  7.  
  8. public void stackLeak(){
  9. stackLength++;
  10. stackLeak();
  11. }
  12.  
  13. public static void main(String[] args) {
  14. StackOFDemo stackOFDemo = new StackOFDemo();
  15. try {
  16. stackOFDemo.stackLeak();
  17. }catch (Throwable e){
  18. System.out.println("length : "+ stackLength);
  19. throw e;
  20. }
  21. }
  22. }

运行结果

  1. length :
  2. Exception in thread "main" java.lang.StackOverflowError
  3. at StackOFDemo.stackLeak(StackOFDemo.java:)
  4. at StackOFDemo.stackLeak(StackOFDemo.java:)
  5. ...

简单解决思路

一般来说此类问题多出现在存在递归的地方,要从代码里重新审视递归未结束的原因,若递归的方法没问题可以根据实际情况调整“-Xss”参数的大小。还有一些代码的循坏依赖也会造成此类情况,

4.直接内存溢出

本机直接内存默认与“-Xmx”设定的值一样大,可以通过“-XX:MaxDirectMemorySize”修改。

  1. /**
  2. * VM Args: -Xmx20m -XX:MaxDirectMemorySize=10
  3. * @author wangzenghuang
  4. */
  5. public class DirectMemoryOOMDemo {
  6. * ;
  7.  
  8. public static void main(String[] args) throws IllegalAccessException {
  9. Field field = Unsafe.];
  10. field.setAccessible(true);
  11. Unsafe unsafe = (Unsafe) field.get(null);
  12. while (true){
  13. unsafe.allocateMemory(_1MB);
  14. }
  15. }
  16. }

运行结果

呃,一运行这段代码idea直接闪退了,查阅其他资料可以得知当DirectMemory导致内存溢出时,Heap Dump文件是很小的,如果程序中有使用NIO的情况可以检查一下。

总结

这里所展示的代码只是可以触发jvm的各种错误,但是并不代表这是唯一的触发错误的方方式,假如我们的代码比较复杂,有时候遇到类似错误的时候还是需要耐心分析。

喜欢的话,关注一下月亮的博客哦,每天分享最干的干货,也可以去我的首页进我的私人群一起讨论、交流、学习!

论JVM爆炸的几种姿势及自救方法,你不得不知!的更多相关文章

  1. 论JVM爆炸的几种姿势及自救方法

    前言 如今不管是在面试还是在我们的工作中,OOM总是不断的出现在我们的视野中,所以我们有必要去了解一下导致OOM的原因以及一些基本的调整方法,大家可以通过下面的事例来了解一下什么样的代码会导致OOM, ...

  2. 【bzoj3224】Tyvj 1728 普通平衡树 01Trie姿势+平衡树的四种姿势 :splay,旋转Treap,非旋转Treap,替罪羊树

    直接上代码 正所谓 人傻自带大常数 平衡树的几种姿势:  AVL Red&Black_Tree 码量爆炸,不常用:SBT 出于各种原因,不常用. 常用: Treap 旋转 基于旋转操作和随机数 ...

  3. 快速了解IOC的几种姿势

    一.首先我们了解IOC如何注入的几种姿势 构造函数注入(Constructor Injection) Ioc容器会智能的选择和调用合适的构造函数以创建依赖的对象.如果被选择的构造函数具有相应的参数,I ...

  4. 两个文件去重的N种姿势

    最近利用shell帮公司优化挖掘关键词的流程,用shell替代了多个环节的操作,极大提高了工作效率. shell在文本处理上确有极大优势,比如多文本合并.去重等,但是最近遇到了一个难搞的问题,即两个大 ...

  5. 又拍云张聪:OpenResty 动态流控的几种姿势

    2019 年 1 月 12 日,由又拍云.OpenResty 中国社区主办的 OpenResty × Open Talk 全国巡回沙龙·深圳站圆满结束,又拍云首席架构师张聪在活动上做了< Ope ...

  6. 补习系列(7)-springboot 实现拦截的五种姿势

    目录 简介 姿势一.使用 Filter 接口 1. 注册 FilterRegistrationBean 2. @WebFilter 注解 姿势二.HanlderInterceptor 姿势三.@Exc ...

  7. python打开文件的N种姿势

    # python打开文件的N种姿势 print('[1]使用open()函数+简单for循环') f1 = open('python.txt') for line in f1: print(line. ...

  8. [转]地图投影的N种姿势

    此处直接给出原文链接: 1.地图投影的N种姿势 2.GIS理论(墨卡托投影.地理坐标系.地面分辨率.地图比例尺.Bing Maps Tile System)

  9. SpringBoot系列教程web篇Servlet 注册的四种姿势

    原文: 191122-SpringBoot系列教程web篇Servlet 注册的四种姿势 前面介绍了 java web 三要素中 filter 的使用指南与常见的易错事项,接下来我们来看一下 Serv ...

随机推荐

  1. How does rt.jar works?

    转载自:https://stackoverflow.com/questions/30222702/how-does-java-link-lib-rt-jar-to-your-app-at-runtim ...

  2. B. Menci 的序列

    题解: 首先subtask1直接状压暴力就好 subtask2我的处理和题解不太一样 仍然正向考虑 设i的时候有最高位为j,那么这个时候数一定越大越好(这个比较好yy) 然后$f[i][j]$搞个高精 ...

  3. 948. Bag of Tokens

    https://leetcode.com/problems/bag-of-tokens/ 一开始觉得应该是个dp 题,把所有结果搜出来然后max 一下.实现以后发现组合太多了,非常慢,即使加上memo ...

  4. hbase 问题整理

    阅读本文可以带着下面问题:1.HBase遇到问题,可以从几方面解决问题?2.HBase个别请求为什么很慢?你认为是什么原因?3.客户端读写请求为什么大量出错?该从哪方面来分析?4.大量服务端excep ...

  5. MongoDb安装和快速入门

    1.Mongodb安装 2.mongodb的增删改查 3.MongoDB数据类型 4.Mongodb $关键字 $修改器 5.MongoDB 之 "$" 的奇妙用法 6.Mongo ...

  6. CTSC2018 被屠记

    占坑 day-1 Gery 和 lll 在火车上谈笑风生 day0 上午 Gery:"我要吊打全场" 下午 Gery 忘带杯子了. Gery:"我过两天碰杯了就可以喝到水 ...

  7. [SCOI2015]小凸玩矩阵

    Description: 给你一个n*m的网格,每个格子有一个数字,每行每列只能选一个数字,问所选数字中第k大的数字的最小值是多少 Hint: \(n \le 250\) Solution: 显然是二 ...

  8. oracle基础语句练习

    1. 创建相关表结构 Emp----员工信息表 Ename ), --姓名 Empno ), --编号 Deptno ), --所在部门 Job ), --工种(人员类别),如:manager 经理, ...

  9. Django之csrf防御机制

    1.csrf攻击过程 csrf攻击说明: 1.用户C打开浏览器,访问受信任网站A,输入用户名和密码请求登录网站A; 2.在用户信息通过验证后,网站A产生Cookie信息并返回给浏览器,此时用户登录网站 ...

  10. protobuf是什么?

      ProtoBuf (Google Protocol Buffer)是由google公司用于数据交换的序列结构化数据格式,具有跨平台.跨语言.可扩展特性,类型于常用的XML及JSON,但具有更小的传 ...