Java内存区域与模拟内存区域异常

我把Java的内存区域画了一张思维导图,以及各区域的主要功能。

Java内存区域


模拟Java堆溢出

Java堆用于存储对象实例,只要不断地创建对象并且保证GC ROOTS到对象之间有可达路径避免被回收机制清除,就可以模拟出Java堆溢出。

package hxl.insist.jvm;

import java.util.ArrayList;
import java.util.List;

/**
 * 下面是JVM Args:
 * -Xms20m 堆的最小值 -Xmx20m 堆的最小值  (设置为一样可避免堆自动扩展)
 * -XX:+HeapDumpOnOutOfMemoryError  当虚拟机出现内存溢出异常时,Dump出当前的堆转储快照
 * -XX:HeapDumpPath=E:\eclipseworkspace\UnderStandingTheJVM\hprof  设置生成的堆转储快照的路径
 * @author hanxl
 *
 */
public class HeapOutOfMemory {

    static class StuffObject {
    }

    static List<StuffObject> list = new ArrayList<StuffObject>();

    public static void main(String[] args) {

        Thread thread = new Thread(new Runnable() {
            public void run() {
                createObj();
            }
        }); 

        thread.start();

    }

    private static void createObj() {
        while (true) {
            list.add(new StuffObject());
        }
    }
}

用MemoryAnalyzer分析一下堆转储快照如下图:

堆转储快照

从根元素到内存消耗聚集点的最短路径,可以很清楚的看到整个引用链。

堆转储快照

在上面这张图上,我们可以清楚的看到,这个对象集合中保存了大量内部类StuffObject 对象的引用,就是它导致的内存泄露。


模拟Java虚拟机栈溢出

关于虚拟机栈,在Java虚拟机中规范了两种异常:

  1. 如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常。
  2. 如果虚拟机地扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常。

模拟第一种情况:

package hxl.insist.jvm;

/**
 * 下面是JVM Args:
 * -Xss128k 设置栈容量大小
 * @author hanxl
 */
public class JavaVMStackSOF {

    public void stackLeak() {
        stackLeak();
    }

    public static void main(String[] args) throws Throwable {
        new JavaVMStackSOF().stackLeak();
    }
}

模拟第二种情况:

package hxl.insist.jvm;
/**
 * 下面是JVM Args:
 * -Xss2M 设置栈容量大小
 * @author hanxl
 */
public class JavaVMStackOOM {

    public static void main(String[] args) {
        new JavaVMStackOOM().threadInvokeMethod();
    }

    public void threadInvokeMethod() {
        while (true) {
            Thread thread = new Thread(new Runnable() {
                @Override
                public void run() {
                    infiniteLoop();
                }
            });
            thread.start();
        }
    }

   private void infiniteLoop() {
          while (true)
              ;
   }
}
比较两种情况为什么实现方式不同?

Java虚拟机栈是线程私有的,它的生命同期与线程相同。我们把虚拟机栈比做一个盒子,-Xss是设置盒子的大小,而一个线程只能对应一个盒子。而每个Java方法在执行的时候都会在盒子中创建一个栈帧用于存储局部变量表等一些信息。

所以为了制造出第一种情况下的异常,我们把盒子的大小设置小一点,使用递归不断调用方法,从而撑破盒子;而为了制造出第二种情况下的异常,我们应该把盒子的大小设置小大一点,多创建一些盒子,从而让其无法申请到足够的内存空间。


只要了解上面那张思维导图的内存区域,模拟出其它内存区域异常也很简单。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值