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

大概可以分为两个时期:
当调用 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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容