public class StringTest {
public static void main(String [] args){
String a = "a";
String b = "a";
String c = "aa";
String d = a+b;
System.out.println(a==b);
System.out.println(c==d);
}
}
结果是
ture
false
为什两次的结果不同。
不要不经思考的就说比较应该用String.equal()
== 比较的是内存地址,为什两次内存地址会不同
2010-09-12 04:25
提问者采纳
楼主,这个是+运算符被Java虚拟机编译时不同解释产生的效果 首先,楼主应该明白,String 是不可变字符串序列,String 有常量池,所有 ""下的字符串序列都是String字符串常量 例如"abc","123" 其次,楼主知道了==比较地址,equals比较值,这个很好,这里就不再罗嗦了 最后,就是重点,为什么System.out.println(a==b);会输出true System.out.println(c==d);会输出false呢 这是因为,当你这样声明一个字符串的时候String s="123";这时候,s直接指向字符串常量池中的地址,假如你再声明String s1="123"这时候,由于常量池中已经存在"123" 字符串序列,所以Java虚拟机不再另外安排空间来产生字符串常量,而把s1这个引用直接指向"123"所以,他们两个当然地址相同 而如果你是这样声明一个字符串String s1=new String("123");这时候,Java会先在堆中创建一个new String 的空间, s1 的引用指向了这个新空间,而这个空间又指向了常量池里的"123"由于s==s1比较的是他们自己的引用地址,所以,返回是false; 相信楼主已经对上面两种情况有所了解了 下面来分析楼主的疑问: String a = "a"; String b = "a"; String c = "aa"; String d = a+b; 来看a和b和c 这个就是我刚才说的直接指向常量池而D呢,乍一看,貌似和C没什么区别,其实不然,这里a是一个引用,b也是一个引用,在用+号进行字符串拼接的时候,编译器首先把A编译成StringBuffer可变字符串序列,然后调用A的append()方法,把B追加进去,然后在调用toString()方法转成字符串序列返回给D,由于是新生成的空间,所以,D引用所存的地址就不是源字符串常量池中的引用,因此返回false 字符串都有一个获得原值引用的方法intern() 楼主可以把==返回是false的字符串都调用下这个方法,会发现都变true了,这个就是直接把引用指向了常量池 刚睡醒,说得一塌糊涂,希望能帮到楼主=。=
-
提问者评价
-
太谢谢你,你回答的太详细了,相信你以前是学习C++的,内存方面学的这么好 交个朋友吧,我学JSP的新手
评论(3)
13
0
2016-3-15 20:10
l蔺相宇l
l蔺相宇l
2015-12-28 13:41
不要再杀影魔了
不要再杀影魔了
2012-9-6 13:58
lwkhehe
lwkhehe
本文深入探讨了Java中String对象的内存存储方式,特别是通过+运算符连接字符串时的行为差异。揭示了字符串常量池的工作原理及其对内存地址的影响。
1236






