OutOfMemoryError异常系列之方法区溢出和运行时常量溢出池溢出

时间:2022-04-25
本文章向大家介绍OutOfMemoryError异常系列之方法区溢出和运行时常量溢出池溢出,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

按照虚拟机的内存分配,运行时常量池属于方法区,所以今天在这一起讲了,大家都知道1.7的虚拟机规范出来以后,有个很重要的一点就是去永久代。今天我们就来观察一下这个实际影响。

String.intern()是一个Native方法。作用就是如果常量池中有这个String变量,则取出,没有的话就是放进去。并返回其String的引用。

在此我们可以看一下1.6的版本使用一个死循环while(true),不断的将String.valueOf(i++).intern()放入一个list中(i之所以变化,是为了不放入同一个String,不然他就会不断地取旧值,不会溢出),为了尽快见到结果,我们可以设置常量池容量,设置-XX:PermSize和-XX:MaxPermSize即可。最后系统会报OutOfMemoryError:PermGen space错误,(我电脑是1.8的jdk,不想下载 了,所以1.6的只是给大家说说,不写代码了)。从这个运行结果可以看出,运行时常量溢出,后面的这个PerGen space说明运行时常量是方法区的一部分。如果大家使用jdk1.7以上版本。将不会报错,while将一直循环下去

/** * VM args : -XX:PermSize=10M -XX:MaxPerSize=10M * Created by 刘洋 on 2017/9/26 0026. */public class RuntimeConstantPool {    public static void main(String [] args){        List<String> list = new ArrayList<>();        int i = 1;        while (true){            list.add(String.valueOf(i++).intern());            System.out.println(i);        }    }}

实验证实,大家可以自己试试。其实还有一个影响,下面贴代码:

public static void main(String [] args){    String str1 = new StringBuilder("123").append("456").toString();    System.out.print(str1.intern() == str1);    String str2 = new StringBuilder("789").append("0").toString();    System.out.print(str2.intern() == str2);}

这个运行在1.7结果是true false,但是在1.6是false false

至于为什么,大家可以自行理解,今天有点累写不动了,下一篇文章我会给大家详解并说一下方法区异常。