深入理解java虚拟机 摘要(三)--实战:OutOfMemoryError异常

时间:2022-12-27 23:29:43

深入理解java虚拟机 摘要


一、自动内存管理机制


3. 实战:OutOfMemoryError异常

  • Java堆溢出:

    测试代码:

    public class Tests {
    static class Obj{
    }

    public static void main(String[] args) throws >Exception {
    List<Obj> list = new ArrayList<>();
    while (true){
    list.add(new Obj());
    }
    }
    }

    启动参数:

    -Xmx20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError

    将堆的最小值-Xms参数与最大值-Xmx参数设置为一样即可避免堆自动扩展,通过参数-XX:+HeapDumpOnOutOfMemoryError可以让虚拟机在出现内存溢
    出异常时Dump出当前的内存堆转储快照以便事后进行分析

    要解决这个区域的异常,一般的手段是先通过内存映像分析工具
    ideal可使用JProfiler和JMeter插件进行分析

  • 虚拟机栈和本地方法栈溢出

    测试代码:

    public static class JavaVMStackSOF {
    private int stackLength = 1;

    public void stackLeak() {
    stackLength++;
    stackLeak();
    }
    }

    public static void main(String[]args)throws Throwable{
    JavaVMStackSOF oom=new JavaVMStackSOF();
    try{
    oom.stackLeak();
    }catch(Throwable e){
    System.out.println("stack length:"+oom.stackLength);
    throw e;
    }
    }

    启动参数:

    -Xss20m

    对于HotSpot来说,虽然-Xoss参数(设置本地方法栈大小)存在,但实际上是无效的,栈容量只由-Xss参数设定
    如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出*Error异常。

    如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常。

    这里把异常分成两种情况,看似更加严谨,但却存在着一些互相重叠的地方:当栈空间无法继续分配时,到底是内存太小,还是已使用的栈空间太大,其本质上只是对同一件事情的两种描述而已。

    在单个线程下,无论是由于栈帧太大还是虚拟机栈容量太小,当内存无法分配的时候,虚拟机抛出的都是*Error异常。

  • 本机直接内存溢出

    DirectMemory容量可通过-XX:MaxDirectMemorySize指定,如果不指定,则默认与Java堆最大值(-Xmx指定)一样

    测试代码:

    public class Tests {
    private static final Long MB=1024L*1024L;
    public static void main(String[]args)throws Exception{
    Field unsafeField=Unsafe.class.getDeclaredFields()[0];
    unsafeField.setAccessible(true);
    Unsafe unsafe=(Unsafe)unsafeField.get(null);
    while(true){
    unsafe.allocateMemory(MB);
    }
    }
    }

    启动参数:

    -Xss20m

    代码解释:
    越过了DirectByteBuffer类,直接通过反射获取Unsafe实例进行内存分配(Unsafe类的getUnsafe()方法限制了只有引导类加载器才会返回实例,也就是设计者希望只有rt.jar中的类才能使用Unsafe的功能)。因为,虽然使用DirectByteBuffer分配内存也会抛出内存溢出异常,但它抛出异常时并没有真正向操作系统申请分配内存,而是通过计算得知内存无法分配,于是手动抛出异常,真正申请分配内存的方法是unsafe.allocateMemory()。

    由DirectMemory导致的内存溢出,一个明显的特征是在Heap Dump文件中不会看见明显的异常,如果读者发现OOM之后Dump文件很小,而程序中又直接或间接使用了NIO,那就可以考虑检查一下是不是这方面的原因。