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