美文网首页
JVM内存划分 字符串常量池

JVM内存划分 字符串常量池

作者: StephenLau | 来源:发表于2020-03-13 16:05 被阅读0次

    字符串常量池

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

    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内存划分 字符串常量池

          本文链接:https://www.haomeiwen.com/subject/irfrshtx.html