java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > java  OutOfMemoryError 异常

JAVA 聚焦 OutOfMemoryError 异常问题记录

作者:小窦总

在 Java 开发中,内存溢出异常是影响程序稳定性的关键问题,了解其原理和应对方法,对开发者至关重要,这篇文章主要介绍了JAVA聚焦 OutOfMemoryError 异常,需要的朋友可以参考下

在 Java 开发中,内存溢出异常是影响程序稳定性的关键问题。了解其原理和应对方法,对开发者至关重要。

一、Java 堆溢出

原理

Java 堆用于存储对象实例。不断创建对象,且阻止垃圾回收器回收,对象数量超出堆容量时,就会引发堆溢出。

示例代码

// VM Args: -Xmx20m -Xms20m -XX:+HeapDumpOnOutOfMemoryError
public class HeapOOM {
    static class OOMObject {}
    public static void main(String[] args) {
        List<OOMObject> list = new ArrayList<>();
        while (true) {
            list.add(new OOMObject());
        }
    }
}

解决思路

二、虚拟机栈和本地方法栈溢出

原理

示例代码

测试StackOverflowError

// VM Args: -Xss128k
public 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 (Exception e) {
            System.out.println("stack length:" + oom.stackLength);
            throw e;
        }
    }
}

测试大量线程导致内存溢出

// VM Args: -Xss2M
public class JavaVMStackOOM {
    private void dontStop() {
        while (true) {}
    }
    public void stackLeakByThread() {
        while (true) {
            Thread thread = new Thread(() -> dontStop());
            thread.start();
        }
    }
    public static void main(String[] args) throws Throwable {
        JavaVMStackOOM oom = new JavaVMStackOOM();
        oom.stackLeakByThread();
    }
}

解决思路

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

原理

示例代码

方法区溢出测试(借助 CGLib )

// VM Args: -XX:PermSize=10M -XX:MaxPermSize=10M
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class JavaMethodAreaOOM {
    static class OOMObject {}
    public static void main(String[] args) {
        while (true) {
            Enhancer enhancer = new Enhancer();
            enhancer.setSuperclass(OOMObject.class);
            enhancer.setUseCache(false);
            enhancer.setCallback(new MethodInterceptor() {
                @Override
                public Object intercept(Object obj, java.lang.reflect.Method method, Object[] args, MethodProxy proxy) throws Throwable {
                    return proxy.invokeSuper(obj, args);
                }
            });
            enhancer.create();
        }
    }
}

运行时常量池溢出测试(String.intern()

// JDK 6 运行:-XX:PermSize=6M -XX:MaxPermSize=6M
// JDK 7及以上运行:-Xmx6M
public class RuntimeConstantPoolOOM {
    public static void main(String[] args) {
        String str1 = new StringBuilder("计算机").append("软件").toString();
        System.out.println(str1.intern() == str1); 
        String str2 = new StringBuilder("ja").append("va").toString();
        System.out.println(str2.intern() == str2); 
    }
}

解决思路

四、直接内存溢出

原理

直接内存容量由-XX:MaxDirectMemorySize 参数控制,默认与 Java 堆最大值相同。直接或间接使用DirectByteBufferUnsafe 等分配内存超出限制,会引发溢出。

示例代码

// VM Args: -Xmx20M -XX:MaxDirectMemorySize=10M
import sun.misc.Unsafe;
import java.lang.reflect.Field;
public class DirectMemoryOOM {
    private static final int _1MB = 1024 * 1024;
    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(_1MB);
        }
    }
}

解决思路

ld.get(null);
while (true) {
unsafe.allocateMemory(_1MB);
}
}
}

解决思路

- 合理设置`-XX:MaxDirectMemorySize` 参数。
- 排查代码中直接内存分配操作,如 NIO 相关代码,确保内存分配合理。

通过深入理解 Java 内存溢出异常原理,结合具体代码示例和解决思路,开发者能更好地定位和解决内存问题,保障 Java 程序稳定运行。

到此这篇关于JAVA 聚焦 OutOfMemoryError 异常问题记录的文章就介绍到这了,更多相关java OutOfMemoryError 异常内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文