美文网首页
关于String intern()在jvm常量池里发生的二三事

关于String intern()在jvm常量池里发生的二三事

作者: miyakee | 来源:发表于2018-02-12 16:24 被阅读0次

    大概可以分为两个时期:
    当调用 intern() 方法时

    jdk1.7之前:

    常量池是在方法区【永久代里面】的
    检查字符串池里是否存在这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会把字符串添加到字符串池中,然后再返回它的引用。
    总结一下:反正这里返回的是字符串池里地址

    jdk1.7(包括)之后:

    常量池被放入到堆空间中
    jvm只是在常量池记录当前字符串的引用,并返回当前字符串的引用[这里返回的是堆的引用]
    总结一下,如果池子里存在,返回池子里的地址,不存在 在池子里加入指向堆的引用,返回的是堆的地址

            String a="hello2";
            String m=new String("hello")+new String("2");
            System.out.println( m==a);
            System.out.println( m.intern()==a);
    

    结果为 false
    true
    因为m是堆里的地址,而"hello2"是字符串常量池里的地址,执行m.intern()的时候,因为常量池已经有了hello2所以直接返回hello2在常量池的引用,和a是一致的

     String m=new String("hello")+new String("2");
     m.intern();
     String a="hello2";
     System.out.println( m==a);
    

    结果为true
    因为在intern的时候,把m加入到常量池后,返回了该堆中的位置,而"hello2"的时候,去常量池一看,啊,已经有这个引用了,就会直接返回这个引用地址,所以指向的是同一个地址

    关于不同方式生成字符串的情况:

    1.String str = "Hello";

    首先会在堆里新生代的Eden区生成一个"Hello"的实例,还会生成字符串常量池(stringTable维护)里生成一个引用 指向堆里的"Hello" 这时候str的位置是指向这个常量池的。

    2.String str = new String("Hello");

    new关键字会在堆申请一块全新的内存,来创建新对象。
    然后这个str 直接指向了堆内实例。同时如果常量池里没有指向hello的字符串,会生成一个

    3.final修饰的字符串

    也是放在字符串常量池里的 所以其实和直接引号生成的对象是一样的。是编译阶段就确定的值。

    4.关于+号连接的字符串

    如果包含new 的字符串 也是在堆里生成新的实例;
    如果全是由引号连接的,会被加入字符串常量池
    如果是一个String对象【一个变量】 会创建一个临时的StringBuilder对象进行实现拼接操作,用StringBuilder的append()方法拼接完毕,再调用toString()方法返回。在编译阶段无法确定值,只有在程序运行期来动态分配并将连接后的新地址赋给s2。返回的是堆里的地址
    如果是一个final修饰的变量与字符串拼接,因为final修饰的时候在编译阶段就会把这个值写进去,所以等同于两个引号的字符串拼接,返回的是堆里的地址。
    参考
    http://tangxman.github.io/2015/07/27/the-difference-of-java-string-pool/
    https://www.zhihu.com/question/29884421/answer/113785601
    http://blog.csdn.net/seu_calvin/article/details/52291082

    相关文章

      网友评论

          本文标题:关于String intern()在jvm常量池里发生的二三事

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