先看代码,以及运行结果
public static void main(String[] args)
{
System.out.println("对象比较");
String str1=new String("str");
String str2=new String("str");
System.out.println("==比较"+(str1==str2));
System.out.println("equal比较"+str1.equals(str2));
System.out.println("值比较");
String str3="str1";
String str4="str1";
System.out.println("==比较"+(str3==str4));
System.out.println("equal比较"+str3.equals(str4));
}
输出的结果为:
根据打印的可以发现使用equal比较时无论是使用自动装箱来实例化还是用new来实例化,返回的都true,而用==则不一样了,自动装箱来实例化的返回的是true,而用new来实例化的返回的确实false;先不急着解决为什么,先来了解下equals和==的区别,到时候就可以知道答案了
equals方法最初是在所有类的基类Object中进行定义的,源码是
public boolean equals(Object obj)
{
return (this == obj);
}
可以看出这里定义的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;
}
经过重写后 equal 与 == 就有了本质的区别了:
equal:是用来比较两个对象内部的内容是否相等的,由于所有的类都是继承自java.lang.Object类的,所以如果没有对该方法进行覆盖的话,调用
的仍然是Object类中的方法,而Object中的equal方法返回的却是==的判断,因此,如果在没有进行该方法的覆盖后,调用该方法是没有
任何意义的。
==:是用来判断两个对象的地址是否相同,即是否是指相同一个对象。比较的是真正意义上的指针操作。
注意:equals与hashCode的定义必须一致:如果x.equals(y)返回true,那么x.hashcode()就必须与y.hashcode()具有相同的值
补充:
在java枚举类中 public enum Size{SMALL,MEDIUM,LARGE,EXTRA_LARGE}
实际上,这个声明定义的类型是一个类,它刚好有四个实例。
因此,在比较两个枚举类型的值时,永远不要调用equals,而直接使用"=="即可。