首先需要了解一下默认的equals

public boolean equals(Object obj) {
return (this == obj);
}

如果在没有修改equals的情况下,equals和==是等价的。

1. ==

  • 对于基本类型数据变量,“==”的作用是判断左右两边的值是否相等(String除外
  • 对于​​引用类型​​数据变量,"=="的作用是判断左右两边所指向的地址是否一样

1. equals

  • 根据equals()的实现代码可知equals()不适用于基本数据类型之间,对于引用类型的变量则和"=="结果相同
  • 如果对equals()方法进行了重写,那么一般情况下就是对比值是否相等(具体和equals实现有关)

验证:以String为例子

String sa = new String("aa");
String sb = "aa";
String sc = "aa";
System.out.println("sa == sb ? "+ (sa == sb ? "true":"false") + ", sb == sc ? "+(sb == sc ? "true":"false"));
System.out.println("sa.equals(sb) ? "+ (sa.equals(sb) ? "true":"false") + ", sb.equals(sc) ? "+(sb.equals(sc) ? "true":"false"));

结果输出:
sa == sb ? false, sb == sc ? true
sa.equals(sb) ? true, sb.equals(sc) ? true

因为编译器有字符串缓存池,所以sb和sc指向的是同一个地址。即:sb == sc 为true。但是sa指定使用new分配的字符串,不是从缓存池里面获取的,所以sa的指向和sb不同,即sa == sb 为false。

String的equals是先比较地址,后比较内容,因为内容都为“aa”所以,sa.equals(sb) 为true,sb.equals(sc)为true;

public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = length();
if (n == anotherString.length()) {
int i = 0;
while (n-- != 0) {
if (charAt(i) != anotherString.charAt(i))
return false;
i++;
}
return true;
}
}
return false;
}