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 {
}
}