Java中字符串比较的问题

时间:2022-02-28 16:15:58
package com.hxl;

import java.util.Scanner;

public class Test {

public static void main(String[] args) {
String s1
= "Hello";
String s2
= "World";
String s3
= "HelloWorld";
System.out.println(s3
== (s1 + s2));
}

}

【结果】false

package com.hxl;

import java.util.Scanner;

public class Test {

public static void main(String[] args) {
String s1
= "Hello";
String s2
= "World";
String s3
= "HelloWorld";
System.out.println((s1
+ s2).hashCode());
System.out.println(s3.hashCode());
}

}

【结果】

439329280
439329280

package com.hxl;

public class Test {
public static void main(String[] args) {
String s1
= "length: 10";
String s2
= "length: " + s1.length();
String s3
= "length: " + 10;
System.out.println(s1);
System.out.println(s2);
System.out.println(s3);
System.out.println(s1.hashCode());
System.out.println(s2.hashCode());
System.out.println(s3.hashCode());
System.out.println(s1
== s2);
System.out.println(s1
== s3);
}
}

【结果】

length: 10
length: 10
length: 10
1160065963
1160065963
1160065963
false
true

【原因】

       首先要知道==比较的是引用,equals比较的是内容。其次还要知道引用放置在栈内,内容放置在堆内。再其次要知道这里没有使用new所以所有数值都是要进入常量池进行比较。当程序加载的时候常量池中存储的有length: 10、length:、s1.length()、10,其中常量10和length:是没有任何引用指向它们的。而length: 10是有s1引用指向它的。s1.length()也是有引用指向的,只不过没有体现出来,或者说s1.length()可以看成是int i=10;或者int i=s1.length();所以这里的s1.length()这个常量10是有引用的指向的一个常量,所以跟上面那个没有引用指向的10是有区别的,而==恰恰是比较引用的,既然引用都不同了,拼接后的引用s2自然跟s1和s3都不一样。所以s2无论是跟s1比较还是跟s3比较结果都是false。

----------------------------------------------------------------------------------------------------

       当定义s3的时候系统会自动把"length: " + 10 转换成 “length: 10” ,让后看看栈空间里面有没有这个值存在,如果有的话就不会在划分空间来存放,直接引用已经存在的值,所以s1跟s3同时引用的都是"length: 10",但s2里面有s1.length()这个是需要通过计算的,每次计算出来的临时的值都会存放在堆空间里面,所以他们引用的地址是不同的。

 ------------------------------------------------------------------------------------------------------------------------

       s1+s2与"Hello"+"World"是不同的,两个变量做运算结果的类型是未知的,而常量运算结果的类型是可以预测的。

【总结】

字符串如果是变量相加,先在栈内存开辟空间,再进行拼接。

字符串如果是常量相加,先拼接,然后在方法区中的常量池中寻找,如果有就直接返回引用。否则,在此处创建并返回引用。