很多不明真相的javaer喜欢纠结于此问题,特来解释一二。 先看代码
StringBuilder
s1 = new
StringBuilder(); StringBuilder
s2 = new
StringBuilder(); String
s = "abc" ; s1.append(s); s2.append(s); System.out.println( "s1.equals(s2):\t\t"
+ (s1.equals(s2))); System.out.println( "s1
== s2:\t\t"
+ (s1 == s2)); String
s3 = s + "1" ; String
s4 = s + "1" ; System.out.println( "s3
== s4:\t\t"
+ (s3 == s4)); System.out.println( "s3.equals(s4):\t\t"
+ (s3.equals(s4))); Object
obj1 = new
Object(); Object
obj2 = new
Object(); System.out.println( "obj1.equals(obj2):\t"
+ (obj1.equals(obj2))); System.out.println( "obj1
== obj2:\t\t"
+ (obj1 == obj2)); |
执行结果如下
s1.equals(s2):
false s1
== s2: false s3
== s4: false s3.equals(s4):
true obj1.equals(obj2):
false obj1
== obj2: false |
有人说“==比较地址,equals比较内容”,从String的比较来看这是对的,但从上述StringBuilder来看,s1和s2的构造方式一模一样,其内容是相同的,但是equals的结果却是false,为何?
我们有必要看一下Object类的equals方法的实现:
public
boolean
equals(Object obj) { return
( this
== obj); } |
Object中的equals就是用==来比较当前对象和传入的参数的。
再看看String的equals实现:
public
boolean
equals(Object anObject) { if
( this
== anObject) { return
true ; } if
(anObject instanceof
String) { String
anotherString = (String)anObject; int
n = count; if
(n == anotherString.count) { char
v1[] = value; char
v2[] = anotherString.value; int
i = offset; int
j = anotherString.offset; while
(n-- != 0 )
{ if
(v1[i++] != v2[j++]) return
false ; } return
true ; } } return
false ; } |
它去比较内容了。
再看StringBuilder,在其源码里没有发现equals方法,那么它就继承了Object的实现,用==来比较传进来的参数。
我们看到equals是个实例方法(非static),实例方法是可以被子类重写而去实现自己想要的行为的,因此,不能轻易的说equals就是比较内容的,其行为是特定于实现的。但==确实是比较地址的。因为java中不支持(至少现在不支持)运算符重载,我们不能改变==的含义,其行为是固定死的。
记得下次不要说“==比较地址,equals比较内容”这样的话了,如果要说,也在前面加上特定的条件,如“如果比较String”,勿断章取义。