JVM内存划分 字符串常量池

字符串常量池

【字符串常量池】独立于【运行时常量池】

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

你可能感兴趣的:(JVM内存划分 字符串常量池)