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

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 {

    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值