看到這個標題,估計很多人會納悶:連集合類的removeAll方法都用不好還當什麼程式員。
好吧,我承認我確實沒用好這個方法,鄙視我吧。O(∩_∩)O哈!
先貼問題--->
實體類(User):
public class User {</p><p>private String name;<br />private int age;</p><p>//setter and getter<br />public String getName() {<br />return name;<br />}<br />public void setName(String name) {<br />this.name = name;<br />}<br />public int getAge() {<br />return age;<br />}<br />public void setAge(int age) {<br />this.age = age;<br />}</p><p>}
測試集合類(UserList):
import java.util.ArrayList;<br />import java.util.List;</p><p>public class UserList {</p><p>private List<User> subList;<br />private List<User> allList;</p><p>public UserList(){<br />subList=new ArrayList<User>();<br />allList=new ArrayList<User>();</p><p>for(int i=0;i<3;i++){<br />User user=new User();<br />user.setAge(i);<br />user.setName("lyh"+i);<br />subList.add(user);<br />}</p><p>for(int i=0;i<10;i++){<br />User user=new User();<br />user.setAge(i);<br />user.setName("lyh"+i);<br />allList.add(user);<br />}<br />}</p><p>public static void main(String[] args){<br />UserList userList=new UserList();<br />userList.allList.removeAll(userList.subList);//調用removeAll方法</p><p>System.out.println(userList.allList.size());</p><p>}<br />}
諸君認為最後的列印的結果是多少? 7 ?That's wrong !! 結果是10
。
為什麼會這樣哪?難道removeAll方法有問題?
這個就是最近在用到removeAll時遇到的疑問(當然,我是把實際中的問題簡單的抽象出來了)。當時百思不得其解,甚至幼稚的以為是Java的BUG 。殊不知是自己腦袋BUG了 !
原因解析:
先看API
boolean removeAll(Collection<?> c)</p><p>從列表中移除指定 collection 中包含的其所有元素(可選操作)。
沒錯,就是移除子集合包含的元素,那為什麼不是7,而是10 。
再看API說明,移除所包含的其所有元素,注意這個字眼:包含!
因為在執行removeAll方法時,會先對集合元素進行比較,如果元素相等才執行移除操作,說到這,相信很多人都已經明白是怎麼回事了,因為不相等(equals),所以沒有執行移除。
為什麼會出現不相等的情況,實體類屬性明明相同,為什麼會不相等?
因為實體類沒有Override hashCode和equals方法 !而在執行removeAll方法時是通過equals方法來判斷集合元素是否相等的,如果沒有Override equals方法,其預設的仍是比較對象,所以會出現上述問題!
歸根到底,還是基礎沒有掌握牢固,同時也給自己提了個醒,寫實體類時盡量Override hashCode和equals方法,不這樣說不定哪天就會出問題。
問題綜述完畢,當然,這個問題實質上很簡單,只希望給遇到有同樣疑問的朋友一點協助。