请谈谈你对OOM的认识

Java中的OOM问题及内存分配策略
本文探讨了Java中两种常见的内存错误——StackOverflowError和OutOfMemoryError。当GC回收效率低下,CPU持续高负载时,可能导致OutofMemory。此外,介绍了ByteBuffer的堆内存和本地内存分配,过度使用直接内存可能导致本地内存耗尽,引发OOM。同时,线程创建过多超出系统限制也会触发此问题。最后,文章提到了方法区(Metaspace)的作用,存储类信息、常量池、静态变量和编译代码,其使用的是本地内存而非虚拟机内存。

java.lang.StackOverFlowError, java.lang.OutOfMemory 属于Error

java.lang.OutOfMemoryError:GC overhead limit exceedec

GC回收时间过长。超过98%的时间用来做GC并且回收了不到2%的堆内存,连续多次GC都只回收了不到2%的极端情况下才会抛出,假如不抛出GC overhead limit 错误会发生什么情况呢?那就是GC清理的这么点内存很快会再次填满,迫使GC再次执行,这样就形成恶性循环,CPU使用率一直是100%,而GC却没有任何成果

public class OOMDemo {

    public static void main(String[] args) {

    }

    @Test
    public void overHeadLimit() {
        List list = new ArrayList();
        int i = 0;
        try {
            while (true) {
                list.add(new String("" + (++i)).intern());
            }
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("i = " + i);
        } finally {
        }
    }
}

[Full GC (Ergonomics) [PSYoungGen: 1024K->1024K(2048K)] [ParOldGen: 7048K->7048K(7168K)] 8072K->8072K(9216K), [Metaspace: 5198K->5198K(1056768K)], 0.0631266 secs] [Times: user=0.06 sys=0.00, real=0.06 secs] 
[Full GC (Ergonomics) [PSYoungGen: 1024K->0K(2048K)] [ParOldGen: 7054K->914K(7168K)] 8078K->914K(9216K), [Metaspace: 5198K->5198K(1056768K)], 0.0362594 secs] [Times: user=0.02 sys=0.00, real=0.04 secs]

java.lang.OutOfMemoryError: GC overhead limit exceeded

	at com.suanfa.booking.OOMDemo.overHeadLimit(OOMDemo.java:23)

java.lang.OutOfMemoryError:Direct buffer memory

ByteBuffer.allocate() 第一种方式是分配JVM堆内存,属于GC管辖范围,由于需要拷贝所以速度相对较慢
ByteBuffer.allocateDirect() 第二种方式是分配OS本地内存,不属于GC管辖范围,由于不需要内存拷贝所以速度相对较快
但如果不断分配本地内存,堆内存很少使用,那么JVM就不需要执行GC,DirectByteBuffer对象们就不会被回收,这时候堆内存充足,但本地内存可能已经使用光了,再次尝试分配本地内存就会出现OutOfMemoryError。

public class OOMDemo {
    @Test
    public void directBufferMemory() {
        System.out.println("maxDirectMemory: " + VM.maxDirectMemory() / (double) 1024 / 1024 + "MB");
        ByteBuffer bb = ByteBuffer.allocateDirect(1800 * 1024 * 1024);
    }
}
maxDirectMemory: 1796.0MB
[GC (System.gc()) [PSYoungGen: 8001K->1448K(38400K)] 8001K->1456K(125952K), 0.0055599 secs] [Times: user=0.03 sys=0.00, real=0.01 secs] 
[Full GC (System.gc()) [PSYoungGen: 1448K->0K(38400K)] [ParOldGen: 8K->1369K(87552K)] 1456K->1369K(125952K), [Metaspace: 5219K->5219K(1056768K)], 0.0150973 secs] [Times: user=0.03 sys=0.00, real=0.02 secs] 

java.lang.OutOfMemoryError: Direct buffer memory

java.lang.OutOfMemoryError:unable to create new native thread

一个应用创建多个线程,超过系统承载极限
linux 非root默认允许单个进程可以创建的线程数是1024个

java.lang.OutOfMemoryError: Metaspace
Metaspace是方法区在HotSpot的实现,他与永久区最大的区别是:Metaspace并不在虚拟机内存中而是使用本地内存

方法区/永久区/元空间 存放了以下信息:
虚拟机加载的类信息
常量池
静态变量
即时编译后的代码

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值