自反性
滿足條件: a.equals(a) = true
保證成立缸兔;
對稱性
滿足條件:如果 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
對輸出結(jié)果解釋如下:
a.equals(b)
比較時,使用 CaseInsensitiveString
邏輯相等的規(guī)則磕蒲,通過String#equalsIgnoreCase
判斷結(jié)果相同。但是在b.equals(a)
比較時臭蚁,使用String#equals()
進(jìn)行邏輯相等的判斷夸赫。
/*String#equals實現(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
。
正確的實現(xiàn)方案
思路: 如果需要比對的兩個對象的類型都不同戴已,直接返回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