java
String str = "abc"; String str2 = "a"; String str3 = "bc"; System.out.println(str == str2 + str3); //fasle System.out.println(str == "a" + str3); //fasle System.out.println(str == "a" + "bc"); //true
第一个与第二个结果为什么会false?
第三个是2个字符串直接拼接会在常量池中生成新的对象(当对象已存在时候直接返回对象地址),所以str
的指向的对象和字符串拼接结果
所指向的对象是相同的,不知道这种理解是否有问题?
string str = "abc";
string str2 = "a";
string str3 = "bc";
Console.WriteLine(str == str2 + str3); //true
Console.WriteLine(str == "a" + str3); //true
Console.WriteLine(str == "a" + "bc"); //true
然而C#中的结果都为true,这2种语言对string类型的维护上有什么不同?求大神解答
@沙渺 正解
分析一下字节码就知道了。
现在一个个的分析:
ldc表示将int, float或String型常量值从常量池中推送至栈顶.
astore表示将栈顶引用型数值存入指定本地变量
那么
表示 abc 在运行时常量池中间取出,并且将引用赋给str变量。
同理,str1,str2都一样。
那么这就说明一个问题:
类似于String str1="dddd"说明"dddd"字符串存在运行时常量池中,而不是存在堆中间。
然后看s2+s3:
java运算符重载了加法。也就是字符串的加号是通过StringBuilder的append和toString来完成的。查看toString的源码:
很显然是新生成了一个String,引用地址不一样,那么"=="判断当然为false。
然后看
说明,对于这种情况,java编译器做了优化。直接存储在运行时常量池中间。那么当然为true。
那么C#。。。。不知道他们运行机制是咋样的。
java 编译期将"a"+"bc"优化为"abc"了
对于Java:
==
比较变量是否指向同一个引用。str2 + str3
和"a" + str3
必然都是临时生成的对象,所以不等。"a" + "bc"
在编译阶段被优化为"abc"
存放在字符串常量表中,常量表中相同内容的只存一次,所以引用相同,比较相等。==
比较字符串肯定是一个不良实践。Don't do this.对于C#:
string
类的==
运算符(MSDN: String.Equality Operator)==
就确实表示比较字符串的内容。System.Object.ReferenceEquals(a, b)
在这个问题上,C#的实现明显是更优的——常用操作简明,不常用操作啰嗦。从而程序编写上语义自然,不易出错。
http://stackoverflow.com/questions/767372/java-string-equals-versus
http://stackoverflow.com/questions/1659097/why-would-you-use-string-eq...
Java中String不是基本类型是对象啊