按照虚拟机的内存分配,运行时常量池属于方法区,所以今天在这一起讲了,大家都知道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
至于为什么,大家可以自行理解,今天有点累写不动了,下一篇文章我会给大家详解并说一下方法区异常。