JVM内存划分 字符串常量池
2020-03-13 本文已影响0人
StephenLau
字符串常量池
【字符串常量池】独立于【运行时常量池】
Java 6只能增加永久代 -XX:MaxPermSize=1G Java 7可以增加字符串 -XX:StringTableSize
运行时常量池中的字符串字面量若是成员的,则在类加载初始化阶段就使用到了字符串常量池;若是本地的,则在使用到的时候才会使用字符串常量池。其实,“使用常量池”对应的字节码是一个ldc指令,在给String类型的引用赋值的时候会先执行这个指令,看常量池中是否存在这个字符串对象的引用,若有就直接返回这个引用,若没有,就在堆里创建这个字符串对象并在字符串常量池中记录下这个引用。String类的intern()方法还可以在运行期间把字符串放到字符串常量池中。
- 在 jdk1.6,在永久代,并且其中存放的是字符串的实例;
- 在 jdk1.7,之后是在堆内存之中
- jdk1.8 字符串常量池是在本地内存当中,存储的也只是引用
String 类和常量池
字面量会返回字符串常量池的对象(没有则创建),new会创建新的。
intern 在常量池中创建/记录此字符串
public class TestRuntimeConstantPool {
public static void main(String[] args) {
/* 1.编译期生成的各种字面量、符号引用 */
String s1 = "abc"; //先检查字符串常量池中有没有字符串,如没有,则创建一个,然后s指向字符串常量池中的对象,如果有,则直接将s指向已有对象
String s2 = "abc"; //常量池中
String s3 = new String("abc"); //堆中创建一个新的对象
/* true:因为"abc"第一次出现将被放在运行时常量池中,后面再有相关变量需要使用,就直接指向它 */
System.out.println(s1 == s2);
/* false:new会直接在java堆空间中创建对象 */
System.out.println(s1 == s3);
/* 2.运行期间新的常量 */
//如果运行时常量池中已经包含一个等于此 String 对象内容的字符串,则返回常量池中该字符串的引用;如果没有,JDK1.7之前(不包含1.7)的处理方式是在常量池中创建与此 String 内容相同的字符串,并返回常量池中创建的字符串的引用,JDK1.7以及之后的处理方式是在常量池中记录此字符串的引用,并返回该引用。
String s4 = s3.intern();
System.out.println(s1 == s4);
}
}
JDK1.7 及之后版本的 JVM 已经将运行时常量池从方法区中移了出来,在 Java 堆(Heap)中开辟了一块区域存放运行时常量池。
String s1 = new String("abc");这句话创建了几个字符串对象?
将创建 1 或 2 个字符串。如果池中已存在字符串常量“abc”,则只会在堆空间创建一个字符串常量“abc”。如果池中没有字符串常量“abc”,那么它将首先在池中创建,然后在堆空间中创建,因此将创建总共 2 个字符串对象。
验证:
String s1 = new String("abc");// 堆内存的地址值
String s2 = "abc";
System.out.println(s1 == s2);// 输出 false,因为一个是堆内存,一个是常量池的内存,故两者是不同的。
System.out.println(s1.equals(s2));// 输出 true