OutOfMemoryError异常

本文详细解析Java中各种内存溢出的情况,包括Java堆、虚拟机栈、方法区和直接内存溢出的原因及解决方案。通过代码示例演示如何触发不同类型的内存溢出,并提供分析和预防内存泄漏的方法。

Java堆溢出
限制Java堆的大小为20M,不可扩展(堆最小值-Xms参数与最大值-Xmx参数设置为一样,避免堆自动扩展),通过参数-XX:+HeapDumpOnOutOfMemoryError可以让虚拟机在出现内存溢出异常时Dump出当前的内存堆转储快照以便事后分析。
在这里插入图片描述

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

public class Main {
    static class OOMObject{

    }
    public static void main(String[] args) {
        List<OOMObject> list = new ArrayList<OOMObject>();
        while (true){
            list.add(new OOMObject());
        }
    }
}
  • 运行结果
    在这里插入图片描述
  • 常规的处理方法:

通过内存映像分析工具对Dump出来的堆转储快照进行分析。
确认是出现了内存泄漏还是内存溢出
内存泄漏: 通过工具查看泄漏对象到GC Roots的引用链,找到泄漏对象是通过怎样的引用路径、与哪些GC Roots相关联,导致垃圾收集器无法回收。根据泄漏对象的类型信息以及它到GC Roots引用链的信息,一般可以比较准确第定位到这些对象创建的位置,进而找出产生内存泄漏的代码的具体位置。
内存溢出: 内存中的对象确实都必须活着。1.应当检查Java虚拟机的堆参数设置(-Xms,-Xmx),与机器内存相比,是否有上调空间。2.检查代码,是否存在某些对象生命周期过长,持有状态时间过长,存储结构设计不合理等情况,尽量减少程序运行期的内存消耗。

  • JProfiler打开的快照文件
    在这里插入图片描述
    (这里内存泄漏定位还要自己实践一下)
    虚拟机栈和本地方法栈溢出
    在《Java虚拟机规范》中描述了两种异常

1.如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常(在递归结束条件写错时,经常出现这个)。
2.如果虚拟机的栈内存允许动态扩展,当扩展栈容量无法申请到足够的内存时,将抛出OutOfMemoryError异常

测试例1

  • JVM配置
    在这里插入图片描述
  • 具体代码
import java.util.ArrayList;
import java.util.List;

public class Main {
    private int stackLength = 1;
    public void stackLeak(){
        stackLength++;
        stackLeak();
    }

    public static void main(String[] args) {
        Main oom = new Main();
        try{
            oom.stackLeak();
        }catch (Throwable e){
            System.out.println("stack length:"+oom.stackLength);
            throw e;
        }
    }
}
  • 运行结果
    在这里插入图片描述
    测试例2
    -Xss2M
  • 具体代码
import java.util.ArrayList;
import java.util.List;

public class Main {
    private void dontStop(){

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

    public static void main(String[] args) {
        Main oom = new Main();
        oom.stackLeakByThread();
    }
}
  • 运行结果
    在这里插入图片描述
    建立过多线程导致的内存溢出,在不能减少线程数量或者更换其他虚拟机的情况下,只能通过减少最大堆和减少栈容量来换取更多的线程(通过减少内存的手段来解决内存溢出)。

方法区和运行时常量池溢出

直接内存溢出
-Xmx20M -XX:MaxDirectMemorySize=10M

  • 具体代码
    通过反射获取Unsafe示例进行内存分配
import sun.misc.Unsafe;

import java.lang.reflect.Field;

public class Main {
    private static final int _1MB = 1024*1024;
    public static void main(String[] args) throws IllegalAccessException {
        Field unsafeField = Unsafe.class.getDeclaredFields()[0];
        unsafeField.setAccessible(true);
        Unsafe unsafe = (Unsafe) unsafeField.get(null);
        while(true){
            unsafe.allocateMemory(_1MB);
        }
    }
}
  • 运行结果
    在这里插入图片描述

参考资料
《深入理解Java虚拟机(第三版)》

参考资源链接:[Java虚拟机内存区域详解:线程私有与共享](https://wenku.youkuaiyun.com/doc/1mk4o498g1?utm_source=wenku_answer2doc_content) 为了避免在Java虚拟机中出现OutOfMemoryError异常,首先需要理解JVM内存区域的结构以及各区域的作用。程序计数器、虚拟机栈、本地方法栈、堆和方法区,这些区域中,堆和方法区是线程共享的,而其他区域则是线程私有的。每种内存区域都有其特定的用途和可能出现的内存溢出异常。 对于程序计数器,由于其内存空间很小并且是线程私有的,所以几乎不会出现OutOfMemoryError异常。但是,当虚拟机栈或本地方法栈中栈帧过多或栈帧过大时,可能会导致StackOverflowError或OutOfMemoryError。为了减少这种风险,需要确保程序中不要创建过多的线程,并且避免方法调用过深或方法内局部变量过大。 在堆内存方面,这是最容易出现OutOfMemoryError的地方。为了避免这种情况,开发者应该注意以下几点: 1. 优化对象的创建和回收,避免产生过多的短期对象。 2. 使用合理的数据结构来减少内存占用。 3. 监控堆内存使用情况,及时调整堆的初始大小和最大大小。 4. 合理配置垃圾回收器,比如选择适合应用特点的垃圾回收算法。 5. 避免循环引用,及时清理不再使用的对象。 方法区的OutOfMemoryError通常与类信息过多有关,比如加载了大量的类或者有过多的常量池信息。为了防止这种情况,应该: 1. 检查代码中是否有大量的类动态生成或者加载。 2. 减少常量池中的数据量,例如避免在运行时生成大量的字符串常量。 3. 关注类加载器的内存使用,避免类加载器泄露。 通过以上措施,可以在项目实践中有效避免不同内存区域导致的OutOfMemoryError异常。更多关于JVM内存区域以及内存管理的详细信息,可以参考《Java虚拟机内存区域详解:线程私有与共享》,该资料将为你提供深入的理解和具体的解决方案。 参考资源链接:[Java虚拟机内存区域详解:线程私有与共享](https://wenku.youkuaiyun.com/doc/1mk4o498g1?utm_source=wenku_answer2doc_content)
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值