概述
在 JDK 1.7 之前
如果要向运行时常量池中添加内容,最简单的做法就是使用 String.intern()这个 Native 方法。该方法的作用是:如果池中已经包含一个等于此 String 对象的字符串,则返回代表池中这个字符串的String 对象;否则,将此 String 对象包含的字符串添加到常量池中,并且返回此 String 对象的引用。由于常量池分配在方法区内,我们可以通过-XX:PermSize 和-XX:MaxPermSize 限制方法区的大小,从而间接限制其中常量池的容量代码运行时常量池导致的内存溢出异常
public class RuntimeConstantPoolOOM {
public static void main(String[] args) {
// 使用 List 保持着常量池引用,避免 Full GC 回收常量池行为
List<String> list = new ArrayList<String>();
// 10MB 的 PermSize 在 integer 范围内足够产生 OOM 了
int i = 0;
while (true) {
list.add(String.valueOf(i++).intern());
}
}
}
运行结果:
Exception in thread "main" java.lang.OutOfMemoryError: PermGen space
at java.lang.String.intern(Native Method)
从运行结果中可以看到,运行时常量池溢出,在 OutOfMemoryError 后面跟随的提示信息是“PermGenspace”,说明运行时常量池属于方法区( HotSpot 虚拟机中的永久代)的一部分。
在JDK 1.7 之后 如下 java 8 中
当常量池中不存在"abc"这个字符串的引用,在堆内存中new一个新的String对象,将这个对象的引用加入常量池。(跟1.6的区别是常量池不再存放对象,只存放引用。)
当常量池中存在"abc"这个字符串的引用,str指向这个引用;
(1)现在当有人问 String str = new String(“abc”);创建了几个对象,常量池有abc字段是1个,常量池没有"abc"字段则是2个。
(2)String str=“abc”;创建了几个对象(如果常量池里面已经有对象了就是0个。如果没有就是1个);
(3)new String(“abc”).intern();创建了几个对象(如果常量池里面已经有该字符串对象了就是1个,如果没有就是两个)
最后
以上就是留胡子流沙为你收集整理的Java虚拟机OOM之运行时常量池溢出的全部内容,希望文章能够帮你解决Java虚拟机OOM之运行时常量池溢出所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复