剖析Java中在Collection集合中使用contains和remove為什么要重寫equals
引言
在Collection集合中:
contains方法是判斷一個集合里面是否包含指定元素,如果有則返回true;
remove方法是從集合中刪除指定元素的單個實(shí)例;
這兩個方法看起很簡單,用起來也很簡單,同樣也非常常用;但是,它們到底是怎么匹配到相應(yīng)的元素呢?
源碼剖析
以ArrayList為例,我們分析一下ArrayList中的contains和remove的源碼;
先看看contains:

這里看到比較的對象是一個Object類,變量名為 o(就是是否包含 o ),并且調(diào)用了一個indexOf方法,接下來我們進(jìn)一步看看indexOf源碼:

可以看到,indexOf又進(jìn)一步調(diào)用了indexOfRange方法,我們還需要深入看看這個方法:

這里可以發(fā)現(xiàn),indexOfRange中 o 調(diào)用了equals方法(藍(lán)色部分)!
我們知道:equals方法是判斷兩個對象是否相等,但是默認(rèn)情況下比較的是對象的地址,如果想要比較對象的內(nèi)容就需要重寫equals方法;
那么這個contains調(diào)用了equals方法,所以,contains判斷一個集合中是否包含某個元素其實(shí)就是通過對象地址比較的了;
這并不是我們想要的結(jié)果,所以幾乎所有放在集合中的類型,都需要重寫equals方法!
為什么是幾乎所有?
因?yàn)檫€是有特例的:SUN公司已經(jīng)把String類和包裝類的equals方法重寫了,所以對于這兩種我們不需要重寫equals!
同樣看看remove方法:

同樣,remove方法也是通過equals方法比較元素然后移除的;
所以這里可以得出一個結(jié)論:
Collection集合中的remove方法和contains方法底層都會調(diào)用equals,所以只要放在集合中的類型,都要重寫equals方法;
因?yàn)閷ο蟮牡刂返谋容^沒有什么意義,我們實(shí)際上需要的是對象內(nèi)容間的比較;
實(shí)例測試
知道了結(jié)論,就來寫幾個代碼測試一下:
String類和包裝類的特殊情況
對于 String類型和包裝類,SUN公司重寫了equals方法,所以我們先測試一下這兩種情況:
import java.util.ArrayList;
import java.util.Collection;
// 結(jié)論:Collection接口中的remove方法和contains方法底層都會調(diào)用equals,
// 所以存放在一個集合中的類型,要重寫它的equals方法
// (但是String和包裝類的equals方法已經(jīng)重寫過了,不用重寫)
public class CollectionTest02 {
public static void main(String[] args) {
// 這里以ArrayList為例
Collection array1 = new ArrayList();
// String類:
String s1 = "Hello";
// 將s1放入array1
array1.add(s1);
// 定義一個s2也為 "Hello",那么調(diào)用contains是否會包含s2?
String s2 = "Hello";
System.out.println("array1是否包含s2?" + array1.contains(s2)); // true
// 因?yàn)閍rray1中放入的是s1,如果移除s2,s1會不會被移除呢?
array1.remove(s2);
System.out.println("移除s2后s1是否還在array1中?" + array1.contains(s1)); // false
// 包裝類也同樣:
Integer num1 = 1000;
// 將num1放入array1
array1.add(num1);
// 定義一個num2也為1000
Integer num2 = 1000;
System.out.println("array1是否包含num2?" + array1.contains(num2)); // true
// 移除num2觀察num1是否會被移除
array1.remove(num2);
System.out.println("移除num2后num1是否還在array1中?" + array1.contains(num1)); // false
}
}
輸出結(jié)果:
array1是否包含s2?true
移除s2后s1是否還在array1中?false
array1是否包含num2?true
移除num2后num1是否還在array1中?false
自定義類型
這是equals重寫的情況,接下來我自定義一個類型,看看沒有重寫會發(fā)生什么;
import java.util.ArrayList;
import java.util.Collection;
public class CollectionTest03 {
public static void main(String[] args) {
// 還是以ArrayList為例
Collection array = new ArrayList();
// 創(chuàng)建一個User對象u1
User u1 = new User("張三");
// 將u1對象放入array中
array.add(u1);
// 再創(chuàng)建一個User對象,使其內(nèi)容和u1相同都為"張三",那么調(diào)用contains是否會包含該對象呢?
System.out.println("array中是否包含新創(chuàng)建的對象?" + array.contains(new User("張三"))); // false
// 移除一個內(nèi)容為"張三"的新的對象,u1是否會被移除?
array.remove(new User("張三"));
System.out.println("移除后u1是否存在?" + array.contains(u1)); // true
}
}
// 自己定義一個User類
class User {
// 成員變量:姓名
private String name;
// 默認(rèn)構(gòu)造
User() {}
// 有參構(gòu)造:初始化姓名
User(String name) {
this.name = name;
}
}
輸出結(jié)果:
array中是否包含新創(chuàng)建的對象?false
移除后u1是否存在?true
可以看到,我自定義的User方法里沒有重寫equals方法,所以當(dāng)調(diào)用contains和remove時,雖然傳入的對象內(nèi)容和u1的對象內(nèi)容相同都為“張三”,但是實(shí)際上比較的卻是對象的地址;
接下來我重寫User的equals方法,看看結(jié)果如何;
import java.util.ArrayList;
import java.util.Collection;
public class CollectionTest03 {
public static void main(String[] args) {
// 還是以ArrayList為例
Collection array = new ArrayList();
// 創(chuàng)建一個User對象u1
User u1 = new User("張三");
// 將u1對象放入array中
array.add(u1);
// 再創(chuàng)建一個User對象,使其內(nèi)容和u1相同都為"張三",那么調(diào)用contains是否會包含該對象呢?
System.out.println("array中是否包含新創(chuàng)建的對象?" + array.contains(new User("張三"))); // true
// 移除一個內(nèi)容為"張三"的新的對象,u1是否會被移除?
array.remove(new User("張三"));
System.out.println("移除后u1是否存在?" + array.contains(u1)); // false
}
}
// 自己定義一個User類
class User {
// 成員變量:姓名
private String name;
// 默認(rèn)構(gòu)造
User() {}
// 有參構(gòu)造:初始化姓名
User(String name) {
this.name = name;
}
// 重寫equals方法 ,通過name進(jìn)行比較
public boolean equals(Object o) {
if (this == o) return true;
if (!(o instanceof User)) return false;
User user = (User) o;
return name.equals(user.name);
}
}
輸出結(jié)果:
array中是否包含新創(chuàng)建的對象?true
移除后u1是否存在?false
我只是重寫了一個equals方法,其他地方都沒變,結(jié)果完全不同;
所以這就驗(yàn)證了之前的結(jié)論:contains 和 remove 底層實(shí)現(xiàn)都調(diào)用了equals方法;
總結(jié)
其實(shí)這篇文章就是分析一下contains 和 remove 底層實(shí)現(xiàn),主要想說的還是那一句話:
Collection集合中的remove方法和contains方法底層都會調(diào)用equals,所以只要放在集合中的類型,都要重寫equals方法;(String和包裝類除外)
希望各位在Java學(xué)習(xí)中養(yǎng)成好習(xí)慣,要時刻惦記著equals的重寫,不然如果做了一個項(xiàng)目你連錯在哪里都不好找到;
到此這篇關(guān)于剖析在Collection集合中使用contains和remove為什么要重寫equals的文章就介紹到這了,更多相關(guān)Collection 重寫 equals內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Mybatis?Web中的數(shù)據(jù)庫操作方法舉例詳解
Mybatis是一款優(yōu)秀的持久化框架,用于簡化JDBC的開發(fā),下面這篇文章主要給大家介紹了關(guān)于Mybatis?Web中數(shù)據(jù)庫操作方法的相關(guān)資料,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-09-09
詳解Spring 框架中切入點(diǎn) pointcut 表達(dá)式的常用寫法
這篇文章主要介紹了詳解Spring 框架中切入點(diǎn) pointcut 表達(dá)式的常用寫法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-04-04
通過代理類實(shí)現(xiàn)java連接數(shù)據(jù)庫(使用dao層操作數(shù)據(jù))實(shí)例分享
java通過代理類實(shí)現(xiàn)數(shù)據(jù)庫DAO操作代碼分享,大家參考使用吧2013-12-12
Springboot2.x+ShardingSphere實(shí)現(xiàn)分庫分表的示例代碼
這篇文章主要介紹了Springboot2.x+ShardingSphere實(shí)現(xiàn)分庫分表的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10

