成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java集合框架總結(jié):TreeSet類的排序問題

開發(fā) 后端
本文介紹TreeSet支持兩種排序方法:自然排序和。TreeSet默認采用自然排序。詳細請看下文

TreeSet支持兩種排序方法:自然排序和定制排序。TreeSet默認采用自然排序。

1、自然排序

TreeSet會調(diào)用集合元素的compareTo(Object obj)方法來比較元素之間大小關(guān)系,然后將集合元素按升序排列,這種方式就是自然排序。(比較的前提:兩個對象的類型相同)。

 

java提供了一個Comparable接口,該接口里定義了一個compareTo(Object obj)方法,該方法返回一個整數(shù)值,實現(xiàn)該接口的類必須實現(xiàn)該方法,實現(xiàn)了該接口的類的對象就可以比較大小。當一個對象調(diào)用該方法與另一個對象進行比較,例如obj1.comparTo(obj2),如果該方法返回0,則表明這兩個對象相等;如果返回一個正整數(shù),則表明obj1大于obj2;如果該方法返回一個負整數(shù),則表明obj1小于obj2.

 

java常用類實現(xiàn)Comparable接口,并提供了比較大小的標準。實現(xiàn)Comparable接口的常用類:

  • BigDecimal、BigIneger以及所有數(shù)值型對應(yīng)包裝類:按它們對應(yīng)的數(shù)值的大小進行比較。
  • Character:按字符的UNICODE值進行比較。
  • Boolean:true對應(yīng)的包裝類實例大于false對應(yīng)的包裝類實例。
  • String:按字符串中字符的UNICODE值進行比較。
  • Date、Time:后面的時間、日期比前面的時間、日期大。

如果試圖把一個對象添加進TreeSet時,則該對象的類必須實現(xiàn)Comparable接口。

如下程序則會報錯:

  1. class Err  
  2. {  
  3. }  
  4. public class TestTreeSetError  
  5. {  
  6. public static void main(String[] args)  
  7. {  
  8. TreeSet ts = new TreeSet();  
  9. //向TreeSet集合中添加兩個Err對象  
  10. ts.add(new Err());  
  11. ts.add(new Err());  
  12. }  
  13. }  
  14.  

 

說明:

上面程序試圖向TreeSet集合中添加2個Err對象,添加第一個對象時,TreeSet里沒有任何元素,所以沒有問題;當添加第二個Err對象時,TreeSet就會調(diào)用該對象的compareTo(Object obj)方法與集合中其他元素進行比較——如果對應(yīng)的類沒有實現(xiàn)Comparable接口,則會引發(fā)ClassCastException異常。而且當試圖從TreeSet中取出元素第一個元素時,依然會引發(fā)ClassCastException異常。

 

當采用compareTo(Object obj)方法比較對象時,都需要將被比較對象obj強制類型轉(zhuǎn)換成相同類型,因為只有相同類的兩個實例才能比較大小。即向TreeSet中添加的應(yīng)該是同一個類的對象,否則會引發(fā)ClassCastException異常。例如,當向TreeSet中添加一個字符串對象,這個操作完全正常。當添加第二個Date對象時,TreeSet就好調(diào)用該對象的compareTo(Object obj)方法與集合中其他元素進行比較,則此時程序會引發(fā)異常。

在實際編程中,程序員可以定義自己的類向TreeSet中添加多種類型的對象,前提是用戶自定義類實現(xiàn)了Comparable接口,實現(xiàn)該接口時在實現(xiàn)compareTo(Object obj)方法時沒有進行強制類型轉(zhuǎn)換。但當操作TreeSet里的集合數(shù)據(jù)時,不同類型的元素依然會發(fā)生ClassCastExceptio異常。(認真閱讀下就會明白)

 

當把一個對象加入TreeSet集合中時,TreeSet調(diào)用該對象的compareTo(Object obj)方法與容器中的其他對象比較大小,然后根據(jù)紅黑樹算法決定它的存儲位置。如果兩個對象通過compareTo(Object obj)比較相等,TreeSet即認為它們存儲同一位置。

 

對于TreeSet集合而言,它判斷兩個對象不相等的標準是:兩個對象通過equals方法比較返回false,或通過compareTo(Object obj)比較沒有返回0——即使兩個對象時同一個對象,TreeSet也會把它們當成兩個對象進行處理。

如下程序所示:

  1. //Z類,重寫了equals方法,總是返回false,  
  2. //重寫了compareTo(Object obj)方法,總是返回正整數(shù)  
  3. class Z implements Comparable  
  4. {  
  5. int age;  
  6. public Z(int age)  
  7. {  
  8. this.age = age;  
  9. }  
  10. public boolean equals(Object obj)  
  11. {  
  12. return false;  
  13. }  
  14. public int compareTo(Object obj)  
  15. {  
  16. return 1;  
  17. }  
  18. }  
  19. public class TestTreeSet  
  20. {  
  21. public static void main(String[] args)  
  22. {  
  23. TreeSet set = new TreeSet();  
  24. Z z1 = new Z(6);  
  25. set.add(z1);  
  26. System.out.println(set.add(z1));  
  27. //下面輸出set集合,將看到有2個元素  
  28. System.out.println(set);  
  29. //修改set集合的第一個元素的age屬性  
  30. ((Z)(set.first())).age = 9;  
  31. //輸出set集合的最后一個元素的age屬性,將看到也變成了9  
  32. System.out.println(((Z)(set.last())).age);  
  33. }  
  34. }  

程序運行結(jié)果:

true 
[TreeSet.Z@1fb8ee3, TreeSet.Z@1fb8ee3] 
9

說明:

程序中把同一個對象添加了兩次,因為z1對象的equals()方法總是返回false,而且compareTo(Object obj)方法總是返回1。這樣TreeSet會認為z1對象和它自己也不相同,因此TreeSet中添加兩個z1對象。而TreeSet對象保存的兩個元素實際上是同一個元素。所以當修改TreeSet集合里第一個元素的age屬性后,該TreeSet集合里最后一個元素的age屬性也隨之改變了。

 

總結(jié):當需要把一個對象放入TreeSet中時,重寫該對象對應(yīng)類的equals()方法時,應(yīng)保證該方法與compareTo(Object obj)方法有一致結(jié)果,其規(guī)則是:如果兩個對象通過equals方法比較返回true時,這兩個對象通過compareTo(Object obj)方法比較應(yīng)返回0。

 

如果兩個對象通過equals方法比較返回true,但這兩個對象通過compareTo(Object obj)方法比較不返回0時,這將導(dǎo)致TreeSet將會把這兩個對象保存在不同位置,從而兩個對象都可以添加成功,這與Set集合的規(guī)則有點出入。

 

如果兩個對象通過compareTo(Object obj)方法比較返回0時,但它們通過equals方法比較返回false時將更麻煩:因為兩個對象通過compareTo(Object obj)方法比較相等,TreeSet將試圖把它們保存在同一個位置,但實際上又不行(否則將只剩下一個對象),所以處理起來比較麻煩。

 

如果向TreeSet中添加一個可變對象后,并且后面程序修改了該可變對象的屬性,導(dǎo)致它與其他對象的大小順序發(fā)生改變,但TreeSet不會再次調(diào)整它們的順序,甚至可能導(dǎo)致TreeSet中保存這兩個對象,它們通過equals方法比較返回true,compareTo(Object obj)方法比較返回0.

如下程序所示:

  1. class R  
  2. {  
  3. int count;  
  4. public R(int count)  
  5. {  
  6. this.count = count;  
  7. }  
  8. public String toString()  
  9. {  
  10. return "R(count屬性:" + count + ")";  
  11. }  
  12. public boolean equals(Object obj)  
  13. {  
  14. if (obj instanceof R)  
  15. {  
  16. R r = (R)obj;  
  17. if (r.count == this.count)  
  18. {  
  19. return true;  
  20. }  
  21. }  
  22. return false;  
  23. }  
  24. public int hashCode()  
  25. {  
  26. return this.count;  
  27. }  
  28. }  
  29. public class TestHashSet2  
  30. {  
  31. public static void main(String[] args)  
  32. {  
  33. HashSet hs = new HashSet();  
  34. hs.add(new R(5));  
  35. hs.add(new R(-3));  
  36. hs.add(new R(9));  
  37. hs.add(new R(-2));  
  38. //打印TreeSet集合,集合元素是有序排列的  
  39. System.out.println(hs);  
  40. //取出第一個元素  
  41. Iterator it = hs.iterator();  
  42. R first = (R)it.next();  
  43. //為第一個元素的count屬性賦值  
  44. first.count = -3;  
  45. //再次輸出count將看到TreeSet里的元素處于無序狀態(tài)  
  46. System.out.println(hs);  
  47. hs.remove(new R(-3));  
  48. System.out.println(hs);  
  49. //輸出false  
  50. System.out.println("hs是否包含count為-3的R對象?" + hs.contains(new R(-3)));  
  51. //輸出false  
  52. System.out.println("hs是否包含count為5的R對象?" + hs.contains(new R(5)));  
  53.  
  54. }  
  55. }  
  56.  

 

程序運行結(jié)果:

[R(count屬性:-3), R(count屬性:-2), R(count屬性:5), R(count屬性:9)]
[R(count屬性:20), R(count屬性:-2), R(count屬性:5), R(count屬性:-2)]
[R(count屬性:20), R(count屬性:-2), R(count屬性:5), R(count屬性:-2)]
[R(count屬性:20), R(count屬性:-2), R(count屬性:-2)]

 

 

說明:

上面程序中的R對象是一個正常重寫了equals方法和comparable方法類,這兩個方法都以R對象的count屬性作為判斷的依據(jù)。可以看到程序第一次輸出的結(jié)果是有序排列的。當改變R對象的count屬性,程序的輸出結(jié)果也發(fā)生了改變,而且包含了重復(fù)元素。一旦改變了TreeSet集合里可變元素的屬性,當再視圖刪除該對象時,TreeSet也會刪除失?。ㄉ踔良现性械?、屬性沒被修改,但與修改后元素相等的元素也無法刪除),所以刪除count

為-2的R對象時,沒有任何元素被刪除;程序可以刪除count為5的R對象,這表明TreeSet可以刪除沒有被修改屬性、且不與其他被修改屬性的對象重復(fù)的對象。

 

總結(jié):與HashSet在處理這些對象時將非常復(fù)雜,而且容易出錯。為了讓程序更具健壯,推薦HashSet和TreeSet集合中只放入不可變對象。

#p#

2、定制排序

TreeSet的自然排序是根據(jù)集合元素的大小,TreeSet將他們以升序排列。如果需要實現(xiàn)定制排序,例如降序,則可以使用Comparator接口。該接口里包含一個int compare(T o1, T o2)方法,該方法用于比較o1和o2的大小。

如果需要實現(xiàn)定制排序,則需要在創(chuàng)建TreeSet集合對象時,并提供一個Comparator對象與該TreeSet集合關(guān)聯(lián),由該Comparator對象負責集合元素的排序邏輯。

如下程序所示:

  1. class M {  
  2. int age;  
  3.  
  4. public M(int age) {  
  5. this.age = age;  
  6. }  
  7.  
  8. public String toString() {  
  9. return "M對象(age:" + age + ")";  
  10. }  
  11. }  
  12.  
  13. public class TestTreeSet3 {  
  14. public static void main(String[] args) {  
  15. TreeSet ts = new TreeSet(new Comparator() {  
  16. public int compare(Object o1, Object o2) {  
  17.  
  18. M m1 = (M) o1;  
  19. M m2 = (M) o2;  
  20.  
  21. if (m1.age > m2.age) {  
  22. return -1;  
  23. else if (m1.age == m2.age) {  
  24. return 0;  
  25. else {  
  26. return 1;  
  27. }  
  28. }  
  29. });  
  30. ts.add(new M(5));  
  31. ts.add(new M(-3));  
  32. ts.add(new M(9));  
  33. System.out.println(ts);  
  34. }  
  35. }  
  36.  

程序運行結(jié)果:

[M對象(age:9), M對象(age:5), M對象(age:-3)]

說明:

上面程序中創(chuàng)建了一個Comparator接口的匿名內(nèi)部類對象,該對象負責ts集合的排序。所以當我們把M對象添加到ts集合中時,無須M類實現(xiàn)Comparable接口,因為此時TreeSet無須通過M對象來比較大小,而是由與TreeSet關(guān)聯(lián)的Comparator對象來負責集合元素的排序。使用定制排序時,TreeSet對集合元素排序時不管集合元素本身的大小,而是由Comparator對象負責集合元素的排序規(guī)則。

原文鏈接:http://www.cnblogs.com/zhxxcq/archive/2012/04/09/2439056.html

【編輯推薦】

  1. 詳解Java類的生命周期
  2. Java理論與實踐: Web層的狀態(tài)復(fù)制
  3. Java代碼編寫的30條建議
  4. Java Excel API及詳細教程
  5. Java集合框架和數(shù)組的排序
責任編輯:林師授 來源: 星星傳奇的博客
相關(guān)推薦

2012-03-12 15:36:29

Java框架

2012-04-26 10:52:52

Java數(shù)組集合

2019-07-22 09:59:20

Java框架集合

2012-03-15 11:21:19

Java

2009-06-30 14:01:00

Java集合框架Java數(shù)組排序

2023-12-01 11:13:50

JavaTreeSet

2015-09-11 09:40:35

Java集合框架

2021-08-19 10:30:13

Java集合排序程序開發(fā)

2011-04-20 15:20:03

快速排序

2011-04-20 15:06:44

堆排序

2011-04-20 13:56:08

選擇排序

2011-04-20 14:07:37

冒泡排序

2011-04-20 14:19:00

希爾排序

2011-07-11 11:02:12

JAVA集合框架

2023-10-09 07:57:14

JavaJCF

2009-06-29 16:50:27

Java集合框架

2021-03-11 07:27:22

Java 集合數(shù)據(jù)

2011-04-20 16:05:15

基數(shù)排序

2011-04-20 12:49:44

插入排序

2011-04-20 14:29:07

歸并排序
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 91一区二区在线观看 | 欧美在线国产精品 | 成年人网站在线观看视频 | 精品国产91 | 懂色中文一区二区三区在线视频 | 一区二区三区欧美 | 欧美日韩精品在线免费观看 | 中文字幕91 | 观看av | 国产精品视频999 | 在线a视频网站 | 日本免费在线看 | 亚洲国产一区二区视频 | 日本成人福利视频 | 亚洲国产一区视频 | 一级毛片色一级 | 成人三级影院 | 国产精品美女久久久久久免费 | 国产精品日日夜夜 | 日本精品在线播放 | 国产美女一区二区三区 | 精品在线观看入口 | 久久精品国内 | 在线一区视频 | 日韩在线不卡视频 | 91综合网 | 狠狠干狠狠插 | 久久91av | 欧美aa在线 | 国产精品成人一区二区 | 国产一区二 | 成人网址在线观看 | 成人在线黄色 | www.国产视频| 久久久av | 日韩成人一区 | 久久久国产精品入口麻豆 | 亚洲视频国产视频 | 黄色综合| 亚洲精品一区二区在线观看 | 欧美成人一级 |