1.编译期的优化
public class Test {
public static void main(String[] args){
String s1="a";
String s2="b";
String s3="ab";
String s4="a"+"b";//在编译时就能确认为"ab"
String s5=s1+s2;//new StringBuffer().append("a").append("b").toString
System.out.println(s3==s4);//true
System.out.println(s3==s5);//false
}
1.String s4="a"+"b"; 由于"a","b"在编译时系统可以肯定其结果为"ab",所以在加载进内存时,直接寻找常量池是否存在"ab",如果有直接指向这块地址 2.String s5=s1+s2; 有一个变量s2,无法确定,它会调用StringBuffer.append("a"),append("b")添加元素。而StringBuffer的toString方法返回的是:return new String("ab");由此可见生成了一个新的引用类型对象,引用类型对象存储在堆中。所以s5指向的是堆中的地址,s3指向常量池中地址
2.intern()方法
首先要知道intern()在jvm中是如何工作的。如s.intern(),先看常量池中是否有与s相同的值,如果没有,就将s加入常量池并返回其对象的引用;如果有则不会将s加入常量池,但还是会返回其常量池中 对象的引用(与s相同的值)。
String s1=new String("ab");
String s2=s1.intern();
System.out.println(s1==s2);//false
先看s1,会创建2个对象,一个"ab"对象在常量池中,对象new String()在堆中。
执行s1.intern()方法时,在常量池中已经有"ab"了,所以不会在将new String("ab")的"ab"加入常量池,最后将常量池中对象“ab”的地址返回给s2。s1指向的是堆中地址,而s2指向的是常量池中地址,所以s1==s2为false。
再看另一种情况
String s1=new String("a")+new String("b");//new String("ab")
String s2=s1.intern();
System.out.println(s1==s2);//true
new String("ab")是在运行时产生的,存在于堆中,在常量池中还没有"ab"对象。接下来执行s1.intern(),将"ab"放入常量池并将其地址返回给s2和s1。所以s1==s2为true。
再看另一种情况
String s1=new String("a")+new String("b");//new String("ab")
s1.intern();
String s2="ab";
System.out.println(s1==s2);//true
此时s2="ab"去常量池寻找是否有"ab",发现有,所以此时s2指向的是常量池中的"ab",与s1的指向同一地址,所以s1==s2结果为true。