自反性
滿足條件: a.equals(a) = true 保證成立;
對(duì)稱性
滿足條件:如果 a.equals(b) = true 那么 b.equals(a)必須成立
違反規(guī)則案例
/*自定義類**/
public class CaseInsensitiveString {
private final String s;
public CaseInsensitiveString(String s) {
this.s = Objects.requireNonNull(s);
}
@Override
public boolean equals(Object obj) {
if (obj instanceof CaseInsensitiveString) {
return s.equalsIgnoreCase(((CaseInsensitiveString) obj).s);
}
if (obj instanceof String) {
return s.equalsIgnoreCase((String) obj);
}
return false;
}
}
/*main函數(shù)**/
public class equalsTest {
public static void main(String[] args) {
CaseInsensitiveString a = new CaseInsensitiveString("luoyonghui");
String b = "luoyonghui";
System.out.println(a.equals(b));
System.out.println(b.equals(a));
List<CaseInsensitiveString> list = new ArrayList<>();
list.add(a);
System.out.println(list.contains(b));
}
}
輸出日志結(jié)果
true
false
false
對(duì)輸出結(jié)果解釋如下:
a.equals(b) 比較時(shí),使用 CaseInsensitiveString邏輯相等的規(guī)則,通過String#equalsIgnoreCase判斷結(jié)果相同。但是在b.equals(a)比較時(shí),使用String#equals()進(jìn)行邏輯相等的判斷。
/*String#equals實(shí)現(xiàn)代碼如下**/
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
如果不是String類型,直接返回false,所有導(dǎo)致上面的輸出解決。同理,ArrayList#contains最終也是通過Object#equals進(jìn)行邏輯判斷,所以輸出結(jié)果也為false。
正確的實(shí)現(xiàn)方案
思路: 如果需要比對(duì)的兩個(gè)對(duì)象的類型都不同,直接返回false。
@Override
public boolean equals(Object obj) {
if (obj instanceof CaseInsensitiveString) {
if (((CaseInsensitiveString) obj).s.equalsIgnoreCase(s)) {
return true;
}
}
return false;
}
輸出日志結(jié)果
false
false
false