String s1 = "one";
String s2 = "on" + "e";
System.out.println(s1 == s2); // trueString s4 = "on";
String s5 = "e";
String s6 = s4 + s5;
System.out.println(s6 == s2); // false不明白为什么第二个是false,String s6 = s4 + s5; 会产生新的String对象吗?
String s2 = "on" + "e";
System.out.println(s1 == s2); // trueString s4 = "on";
String s5 = "e";
String s6 = s4 + s5;
System.out.println(s6 == s2); // false不明白为什么第二个是false,String s6 = s4 + s5; 会产生新的String对象吗?
String s2 = "on" + "e";这样的会被编译器解释为"one"
还要理解String 池
我刚上传了一个java面试题解惑的pdf,上边有一章专门讲解String创建对象的过程,你看了一定会明白的!免积分下载的。
String s6 = s4 + s5 会重新生成一个对象
所以返回false详细请参考 http://blog.csdn.net/hzc543806053/article/details/7324360
原理1:当使用任何方式来创建一个字符串对象s时,Java运行时(运行中JVM)会拿着这个X在String池中找是否存在内容相同的字符串对象,如果不存在,则在池中创建一个字符串s,否则,不在池中添加。
原理2:Java中,只要使用new关键字来创建对象,则一定会(在堆区或栈区)创建一个新的对象。
原理3:使用直接指定或者使用纯字符串串联来创建String对象,则仅仅会检查维护String池中的字符串,池中没有就在池中创建一个,有则罢了!但绝不会在堆栈区再去创建该String对象。
原理4:使用包含变量的表达式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区创建一个String对象。
我刚上传了一个面试题解惑的pdf,上边有一章是专门讲解string创建对象的,看了你就会明白的!免积分下载的!
String s2 = "on" + "e"; //编译器在这里知道"on"+"e"="one",编译优化,s2引用常量池的"one"
System.out.println(s1 == s2); //trueString s4 = "on";
String s5 = "e";
String s6 = s4 + s5; //编译器在这里不知道s4和s5是什么,只知道s4和s5是两个变量,
//编译器在这里只知道s6是两个变量相加的结果,所以要等到运行期执行才能确定s6,
//而在运行期的两个字符串相加,会调用StringBuilder的toString完成,
//所以s6引用的是堆中"one",和常量池的"one"不是同一个对象(所以引用的地址不同)
System.out.println(s6 == s2); // 所以,这里当然就是false了
谢谢你的回答,对于原理1我有点小疑问:
1.JVM如何去寻找这样内容相同的字符串,也是用equals方法吗?如果不是,它的算法是什么呢?
2.如果程序很大,每次遇到String的都在String池中创建字符串,会不会导致内存占用过多?