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

来源:互联网 发布:linux 线程退出 编辑:程序博客网 时间:2024/06/07 00:33

1. 概述

运行时常量池是方法区(永久代)的一部分
参数-XX:PermSize和-XX:MaxPermSize限制方法区的大小,从而间接限制了常量池的大小。
- JDK6及之前的版本中,字符串常量池在永久代中
- 已发布的JDK7的HotSpot中,已经把原本放在永久代中的字符串常量池移出。

2. 运行时常量池导致的内存溢出异常

String.intern是一个Native方法,作用:
- 如果字符串常量池中已经包含一个等于String对象的字符串,则返回常量池中这个字符串的String对象
- 如果不包含,则将此String对象包含的字符串添加到常量池中,并返回此String对象的引用。

package com.java.one;import java.util.ArrayList;import java.util.List;/** * 运行时常量池导致的内存溢出异常 * VM Args: -XX:PermSize=10M -XX:MaxPermSize=20M * */public class RuntimeConstantPoolOOM {    public static void main(String[] args) {        // 使用List保持着常量池的引用,避免Full GC回收常量池的行为        List<String> list = new ArrayList<>();        int i = 0;        while (true) {            list.add(String.valueOf(i++).intern());        }    }}
  • JDK6及以前,内存溢出OOM,说明运行时常量池属于方法区的一部分
  • JDK7及以后,没有OOM,while循环将一直进行下去

3. String.intern()返回引用的测试

package com.java.one;/** * String.intern()返回引用的测试 * */public class RuntimeConstantStringIntern {    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);    }}

这里写图片描述
JDK8中String.intern()返回的是在字符串常量池中首次出现的实例的引用,字符串“计算机软件”是第一次出现,所以返回true,而”java”之前就出现过,与现在创建的”java”字符串引用已经不同了,返回false。

4. 借助CGLib使方法区出现内存溢出异常

方法区用于存放Class的相关信息,对于这些区域测试的基本思路:运行时产生大量的类去填满方法区,知道溢出。

package com.java.one;import java.lang.reflect.Method;import java.lang.reflect.Proxy;/** * 借助CGLib使方法区出现内存溢出异常 * 借助CGLib直接操作字节码运行时产生大量的动态类 * VM Args: -XX:PermSize=10M -XX:MaxPermSize=10M * */public class JavaMethodAreaOOM {    public static void main(String[] args) {        while (true) {            Enhancer enhancer = new Enhancer();            enhancer.setSuperclass(OOMObject.class);            enhancer.setUseCache(false);            enhancer.setCallBack(new MethodInterceptor() {                public Object intercept (Object obj, Method method, Object[] args, MethodProxy proxy) {                    return proxy.invokeSuper(obj, args);                }            });            enhancer.create();        }    }    static class OOMObject {    }}
原创粉丝点击