先看以下代码:
public class test {
public static void main(String[] args){
String s1 = "a";
String s2 = "b";
String s3 = "ab";
String s4 = s1+s2;
String s5 = "a"+"b";
String s6 = s1+"b";
StringBuffer s7 = new StringBuffer("abc");
String s8 = s7.substring(0,2);
String s9 = "ab";
String s10 = new String("ab");
StringBuffer s11 = new StringBuffer("abc");
System.out.println(s3==s4);
System.out.println(s3 == s5);
System.out.println(s3==s6);
System.out.println(s4==s6);
System.out.println(s4==s8);
System.out.println(s3==s9);
System.out.println(s3==s10);
System.out.println(s7.equals(s11));
System.out.println(s3.equals(s4));
}
}
运行结果:
false
true
false
false
false
true
false
false
true
分析:
1)对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;
如果作用于引用类型的变量,则比较的是所指向的对象的地址
2)对于equals方法,注意:equals方法不能作用于基本数据类型的变量
如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;
诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容。比较的是所指向的对象的内容。
3) 程序在运行的时候会创建一个字符串缓冲池,程序会把字符串常量(如程序中的”a”,”b”,”ab”,”a”+”b”)放入字符串缓冲池中(这里”ab”,”a”+”b”是一个对象)。当使用
String s3 = “ab”;
String s5 = “a” + “b”;
在创建字符串的时候,程序首先会在这个String缓冲池中寻找相同值的对象,s3先被放到了池中,所以在s5被创建的时候,程序找到了具有相同值的 s3,将s5引用s3所引用的对象”ab”。所以比较s5==s3由于两个是指向同一个对象,所以为true。
但是在这些赋值中,涉及到变量运算,所以得到的字符串对象存储在各自的内存中,不是缓冲池中的s3对象,如
String s4 = s1 + s2;
String s6 = s1 + “b”;
所以 尽管s4,s6的值均为”ab”,但是 s4==s3,s6==s3返回的结果均为false;
同时s4,s6也不是指向相同的对象,因为他们在不同内存中。
StringBuffer s7 = new StringBuffer(“abc”);
String s8 = s7.substring(3);
s7 新申请了一个内存空间,所以s8==s3的结果也为false;
s10的new String会创建一个新的对象,这时引用就是新的了。
注:new String()是创建新的内存
s= “aaaaa”;创建到字符串缓冲区中
String 重写了equals方法,使其比较的是值,而StringBuffer没有重写,所以s7.equals(s11)是false,而s3.equals(s4)是true。
参考网址: