自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

深入淺出的分析 Set集合

開發(fā) 前端
關(guān)于 Set 接口,在實(shí)際開發(fā)中,其實(shí)很少用到,但是如果你出去面試,它可能依然是一個(gè)繞不開的話題。

 01. 摘要

“關(guān)于 Set 接口,在實(shí)際開發(fā)中,其實(shí)很少用到,但是如果你出去面試,它可能依然是一個(gè)繞不開的話題。

[[282307]]

言歸正傳,廢話咱們也不多說(shuō)了,相信使用過(guò) Set 集合類的朋友都知道,Set集合的特點(diǎn)主要有:元素不重復(fù)、存儲(chǔ)無(wú)序的特點(diǎn)。

啥意思呢?你可以理解為,向一個(gè)瓶子里面扔?xùn)|西,這些東西沒(méi)有記號(hào)是第幾個(gè)放進(jìn)去的,但是有一點(diǎn)就是這個(gè)瓶子里面不會(huì)有重樣的東西。

細(xì)細(xì)思考,你會(huì)發(fā)現(xiàn), Set 集合的這些特性正處于 List 集合和 Map 集合之間,為什么這么說(shuō)呢?之前的集合文章中,咱們了解到,List 集合的特點(diǎn)就是存取有序,本質(zhì)是一個(gè)有序數(shù)組,每個(gè)元素依次按照順序存儲(chǔ);Map 集合主要用于存放鍵值對(duì),雖然底層也是用數(shù)組存放,但是元素在數(shù)組中的下標(biāo)是通過(guò)哈希算法計(jì)算出來(lái)的,數(shù)組下標(biāo)無(wú)序。

而 Set 集合,在元素存儲(chǔ)方面,注重獨(dú)立無(wú)二的特性,如果某個(gè)元素在集合中已經(jīng)存在,不會(huì)存儲(chǔ)重復(fù)的元素,同時(shí),集合存儲(chǔ)的是元素,不像 Map 集合那樣存儲(chǔ)的是鍵值對(duì)。

具體的分析,咱們慢慢道來(lái),打開 Set 集合,主要實(shí)現(xiàn)類有 HashSet、LinkedHashSet 、TreeSet 、EnumSet( RegularEnumSet、JumboEnumSet )等等,總結(jié) Set 接口實(shí)現(xiàn)類,圖如下:

 

 

 

 

由圖中的繼承關(guān)系,可以知道,Set 接口主要實(shí)現(xiàn)類有 AbstractSet、HashSet、LinkedHashSet 、TreeSet 、EnumSet( RegularEnumSet、JumboEnumSet ),其中 AbstractSet、EnumSet 屬于抽象類,EnumSet 是在 jdk1.5 中新增的,不同的是 EnumSet 集合元素必須是枚舉類型。

  • HashSet 是一個(gè)輸入輸出無(wú)序的集合,集合中的元素基于 HashMap 的 key 實(shí)現(xiàn),元素不可重復(fù);
  • LinkedHashSet 是一個(gè)輸入輸出有序的集合,集合中的元素基于 LinkedHashMap 的 key 實(shí)現(xiàn),元素也不可重復(fù);
  • TreeSet 是一個(gè)排序的集合,集合中的元素基于 TreeMap 的 key 實(shí)現(xiàn),同樣元素不可重復(fù);
  • EnumSet 是一個(gè)與枚舉類型一起使用的專用 Set 集合,其中 RegularEnumSet 和 JumboEnumSet 不能單獨(dú)實(shí)例化,只能由 EnumSet 來(lái)生成,同樣元素不可重復(fù);

下面咱們來(lái)對(duì)各個(gè)主要實(shí)現(xiàn)類進(jìn)行一一分析!

02. HashSet

HashSet 是一個(gè)輸入輸出無(wú)序的集合,底層基于 HashMap 來(lái)實(shí)現(xiàn),HashSet 利用 HashMap 中的key元素來(lái)存放元素,這一點(diǎn)我們可以從源碼上看出來(lái),閱讀源碼如下:

 

  1. public class HashSet<E> 
  2.     extends AbstractSet<E> 
  3.     implements Set<E>, Cloneable, java.io.Serializable
  4.      
  5.     // HashMap 變量 
  6.     private transient HashMap<E,Object> map; 
  7.      
  8.     /**HashSet 初始化*/ 
  9.     public HashSet() { 
  10.         //默認(rèn)實(shí)例化一個(gè) HashMap 
  11.         map = new HashMap<>(); 
  12.     } 

add方法

打開HashSet的add()方法,源碼如下:

 

  1. public boolean add(E e) { 
  2.     //向 HashMap 中添加元素 
  3.     return map.put(e, PRESENT)==null

其中變量PRESENT,是一個(gè)非空對(duì)象,源碼部分如下:

 

  1. private static final Object PRESENT = new Object(); 

可以分析出,當(dāng)進(jìn)行add()的時(shí)候,等價(jià)于

 

  1. HashMap map = new HashMap<>(); 
  2. map.put(e, new Object());//e 表示要添加的元素 

在之前的集合文章中,咱們了解到 HashMap 在添加元素的時(shí)候 ,通過(guò)equals()和hashCode()方法來(lái)判斷傳入的key是否相同,如果相同,那么 HashMap 認(rèn)為添加的是同一個(gè)元素,反之,則不是。

從源碼分析上可以看出,HashSet 正是使用了 HashMap 的這一特性,實(shí)現(xiàn)存儲(chǔ)元素下標(biāo)無(wú)序、元素不會(huì)重復(fù)的特點(diǎn)。

remove方法

HashSet 的刪除方法,同樣如此,也是基于 HashMap 的底層實(shí)現(xiàn),源碼如下:

 

  1. public boolean remove(Object o) { 
  2.     //調(diào)用HashMap 的remove方法,移除元素 
  3.     return map.remove(o)==PRESENT; 

查詢方法

HashSet 沒(méi)有像 List、Map 那樣提供 get 方法,而是使用迭代器或者 for 循環(huán)來(lái)遍歷元素,方法如下:

 

  1. public static void main(String[] args) { 
  2.     Set<String> hashSet = new HashSet<String>(); 
  3.     System.out.println("HashSet初始容量大?。?quot;+hashSet.size()); 
  4.     hashSet.add("1"); 
  5.     hashSet.add("2"); 
  6.     hashSet.add("3"); 
  7.     hashSet.add("3"); 
  8.     hashSet.add("2"); 
  9.     hashSet.add(null); 
  10.  
  11.     //相同元素會(huì)自動(dòng)覆蓋 
  12.     System.out.println("HashSet容量大小:"+hashSet.size()); 
  13.     //迭代器遍歷 
  14.     Iterator<String> iterator = hashSet.iterator(); 
  15.     while (iterator.hasNext()){ 
  16.         String str = iterator.next(); 
  17.         System.out.print(str + ","); 
  18.     } 
  19.  
  20.     System.out.println("\n==========="); 
  21.     //增強(qiáng)for循環(huán) 
  22.     for (String str : hashSet) { 
  23.         System.out.print(str + ","); 
  24.     } 

輸出結(jié)果:

 

  1. HashSet初始容量大?。? 
  2. HashSet容量大?。? 
  3. null,1,2,3, 
  4. =========== 
  5. null,1,2,3, 

需要注意的是,HashSet 允許添加為null的元素。

03. LinkedHashSet

LinkedHashSet 是一個(gè)輸入輸出有序的集合,繼承自 HashSet,但是底層基于 LinkedHashMap 來(lái)實(shí)現(xiàn)。

如果你之前了解過(guò) LinkedHashMap,那么你一定知道,它也繼承自 HashMap,唯一有區(qū)別的是,LinkedHashMap 底層數(shù)據(jù)結(jié)構(gòu)基于循環(huán)鏈表實(shí)現(xiàn),并且數(shù)組指定了頭部和尾部,雖然數(shù)組的下標(biāo)存儲(chǔ)無(wú)序,但是卻可以通過(guò)數(shù)組的頭部和尾部,加上循環(huán)鏈表,依次可以查詢到元素存儲(chǔ)的過(guò)程,從而做到輸入輸出有序的特點(diǎn)。

如果還不了解 LinkedHashMap 的實(shí)現(xiàn)過(guò)程,可以參閱集合系列中關(guān)于 LinkedHashMap 的實(shí)現(xiàn)過(guò)程文章。

閱讀 LinkedHashSet 的源碼,類定義如下:

 

  1. public class LinkedHashSet<E> 
  2.     extends HashSet<E> 
  3.     implements Set<E>, Cloneable, java.io.Serializable { 
  4.  
  5.     public LinkedHashSet() { 
  6.         //調(diào)用 HashSet 的方法 
  7.         super(16, .75f, true); 
  8.     } 

查詢?cè)创a,super調(diào)用的方法,源碼如下:

 

  1. HashSet(int initialCapacity, float loadFactor, boolean dummy) { 
  2.     //初始化一個(gè) LinkedHashMap 
  3.     map = new LinkedHashMap<>(initialCapacity, loadFactor); 

add方法

LinkedHashSet沒(méi)有重寫add方法,而是直接調(diào)用HashSet的add()方法,因?yàn)閙ap的實(shí)現(xiàn)類是LinkedHashMap,所以此處是向LinkedHashMap中添加元素,當(dāng)進(jìn)行add()的時(shí)候,等價(jià)于

 

  1. HashMap map = new LinkedHashMap<>(); 
  2. map.put(e, new Object());//e 表示要添加的元素 

remove方法

LinkedHashSet也沒(méi)有重寫remove方法,而是直接調(diào)用HashSet的刪除方法,因?yàn)長(zhǎng)inkedHashMap沒(méi)有重寫remove方法,所以調(diào)用的也是HashMap的remove方法,源碼如下:

 

  1. public boolean remove(Object o) { 
  2.     //調(diào)用HashMap 的remove方法,移除元素 
  3.     return map.remove(o)==PRESENT; 

查詢方法

同樣的,LinkedHashSet 沒(méi)有提供 get 方法,使用迭代器或者 for 循環(huán)來(lái)遍歷元素,方法如下:

 

  1. public static void main(String[] args) { 
  2.     Set<String> linkedHashSet = new LinkedHashSet<String>(); 
  3.     System.out.println("linkedHashSet初始容量大小:"+linkedHashSet.size()); 
  4.     linkedHashSet.add("1"); 
  5.     linkedHashSet.add("2"); 
  6.     linkedHashSet.add("3"); 
  7.     linkedHashSet.add("3"); 
  8.     linkedHashSet.add("2"); 
  9.     linkedHashSet.add(null); 
  10.     linkedHashSet.add(null); 
  11.  
  12.     System.out.println("linkedHashSet容量大?。?quot;+linkedHashSet.size()); 
  13.     //迭代器遍歷 
  14.     Iterator<String> iterator = linkedHashSet.iterator(); 
  15.     while (iterator.hasNext()){ 
  16.         String str = iterator.next(); 
  17.         System.out.print(str + ","); 
  18.     } 
  19.  
  20.     System.out.println("\n==========="); 
  21.     //增強(qiáng)for循環(huán) 
  22.     for (String str : linkedHashSet) { 
  23.         System.out.print(str + ","); 
  24.     } 

輸出結(jié)果:

 

  1. linkedHashSet初始容量大?。? 
  2. linkedHashSet容量大小:4 
  3. 1,2,3,null
  4. =========== 
  5. 1,2,3,null

可見,LinkedHashSet 與 HashSet 相比,LinkedHashSet 輸入輸出有序。

04. TreeSet

TreeSet 是一個(gè)排序的集合,實(shí)現(xiàn)了NavigableSet、SortedSet、Set接口,底層基于 TreeMap 來(lái)實(shí)現(xiàn)。TreeSet 利用 TreeMap 中的key元素來(lái)存放元素,這一點(diǎn)我們也可以從源碼上看出來(lái),閱讀源碼,類定義如下:

 

  1. public class TreeSet<E> extends AbstractSet<E> 
  2. implements NavigableSet<E>, Cloneable, java.io.Serializable { 
  3.      
  4.     //TreeSet 使用NavigableMap接口作為變量 
  5.     private transient NavigableMap<E,Object> m; 
  6.      
  7.     /**對(duì)象初始化*/ 
  8.     public TreeSet() { 
  9.         //默認(rèn)實(shí)例化一個(gè) TreeMap 對(duì)象 
  10.         this(new TreeMap<E,Object>()); 
  11.     } 
  12.      
  13.     //對(duì)象初始化調(diào)用的方法 
  14.     TreeSet(NavigableMap<E,Object> m) { 
  15.         this.m = m; 
  16.     } 

new TreeSet<>()對(duì)象實(shí)例化的時(shí)候,表達(dá)的意思,可以簡(jiǎn)化為如下:

 

  1. NavigableMap<E,Object> m = new TreeMap<E,Object>(); 

因?yàn)門reeMap實(shí)現(xiàn)了NavigableMap接口,所以沒(méi)啥問(wèn)題。

 

  1. public class TreeMap<K,V> 
  2.     extends AbstractMap<K,V> 
  3.     implements NavigableMap<K,V>, Cloneable, java.io.Serializable
  4.     ...... 

add方法

打開TreeSet的add()方法,源碼如下:

 

  1. public boolean add(E e) { 
  2.     //向 TreeMap 中添加元素 
  3.     return m.put(e, PRESENT)==null

其中變量PRESENT,也是是一個(gè)非空對(duì)象,源碼部分如下:

 

  1. private static final Object PRESENT = new Object(); 

可以分析出,當(dāng)進(jìn)行add()的時(shí)候,等價(jià)于

 

  1. TreeMap map = new TreeMap<>(); 
  2. map.put(e, new Object());//e 表示要添加的元素 

TreeMap 類主要功能在于,給添加的集合元素,按照一個(gè)的規(guī)則進(jìn)行了排序,默認(rèn)以自然順序進(jìn)行排序,當(dāng)然也可以自定義排序,比如測(cè)試方法如下:

 

  1. public static void main(String[] args) { 
  2.     Map initMap = new TreeMap(); 
  3.     initMap.put("4""d"); 
  4.     initMap.put("3""c"); 
  5.     initMap.put("1""a"); 
  6.     initMap.put("2""b"); 
  7.     //默認(rèn)自然排序,key為升序 
  8.     System.out.println("默認(rèn) 排序結(jié)果:" + initMap.toString()); 
  9.     //自定義排序,在TreeMap初始化階段傳入Comparator 內(nèi)部對(duì)象 
  10.     Map comparatorMap = new TreeMap<String, String>(new Comparator<String>() { 
  11.         @Override 
  12.         public int compare(String o1, String o2){ 
  13.             //根據(jù)key比較大小,采用倒敘,以大到小排序 
  14.             return o2.compareTo(o1); 
  15.         } 
  16.     }); 
  17.     comparatorMap.put("4""d"); 
  18.     comparatorMap.put("3""c"); 
  19.     comparatorMap.put("1""a"); 
  20.     comparatorMap.put("2""b"); 
  21.     System.out.println("自定義 排序結(jié)果:" + comparatorMap.toString()); 

輸出結(jié)果:

 

  1. 默認(rèn) 排序結(jié)果:{1=a, 2=b, 3=c, 4=d} 
  2. 自定義 排序結(jié)果:{4=d, 3=c, 2=b, 1=a} 

相信使用過(guò)TreeMap的朋友,一定知道TreeMap會(huì)自動(dòng)將key按照一定規(guī)則進(jìn)行排序,TreeSet正是使用了TreeMap這種特性,來(lái)實(shí)現(xiàn)添加的元素集合,在輸出的時(shí)候,其結(jié)果是已經(jīng)排序好的。

如果您沒(méi)看過(guò)源碼TreeMap的實(shí)現(xiàn)過(guò)程,可以參閱集合系列文章中TreeMap的實(shí)現(xiàn)過(guò)程介紹,或者閱讀 jdk 源碼。

remove方法

TreeSet 的刪除方法,同樣如此,也是基于 TreeMap 的底層實(shí)現(xiàn),源碼如下:

 

  1. public boolean remove(Object o) { 
  2.         //調(diào)用TreeMap 的remove方法,移除元素 
  3.         return m.remove(o)==PRESENT; 

查詢方法

TreeSet 沒(méi)有重寫 get 方法,而是使用迭代器或者 for 循環(huán)來(lái)遍歷元素,方法如下:

 

  1. public static void main(String[] args) { 
  2.     Set<String> treeSet = new TreeSet<>(); 
  3.     System.out.println("treeSet初始容量大?。?quot;+treeSet.size()); 
  4.     treeSet.add("1"); 
  5.     treeSet.add("4"); 
  6.     treeSet.add("3"); 
  7.     treeSet.add("8"); 
  8.     treeSet.add("5"); 
  9.  
  10.     System.out.println("treeSet容量大?。?quot;+treeSet.size()); 
  11.     //迭代器遍歷 
  12.     Iterator<String> iterator = treeSet.iterator(); 
  13.     while (iterator.hasNext()){ 
  14.         String str = iterator.next(); 
  15.         System.out.print(str + ","); 
  16.     } 
  17.  
  18.     System.out.println("\n==========="); 
  19.     //增強(qiáng)for循環(huán) 
  20.     for (String str : treeSet) { 
  21.         System.out.print(str + ","); 
  22.     } 

輸出結(jié)果:

 

  1. treeSet初始容量大小:0 
  2. treeSet容量大?。? 
  3. 1,3,4,5,8, 
  4. =========== 
  5. 1,3,4,5,8, 

自定義排序

使用自定義排序,有 2 種方法,第一種在需要添加的元素類,實(shí)現(xiàn)Comparable接口,重寫compareTo方法來(lái)實(shí)現(xiàn)對(duì)元素進(jìn)行比較,實(shí)現(xiàn)自定義排序。

方法一

 

  1. /** 
  2.   * 創(chuàng)建實(shí)體類Person實(shí)現(xiàn)Comparable接口 
  3.   */ 
  4. public class Person implements Comparable<Person>{ 
  5.     private int age; 
  6.     private String name
  7.     public Person(String nameint age){ 
  8.         this.name = name
  9.         this.age = age; 
  10.     } 
  11.     @Override 
  12.     public int compareTo(Person o){ 
  13.         //重寫 compareTo 方法,自定義排序算法 
  14.         return this.age-o.age; 
  15.     } 
  16.     @Override 
  17.     public String toString(){ 
  18.         return name+":"+age; 
  19.     } 

創(chuàng)建一個(gè)Person實(shí)體類,實(shí)現(xiàn)Comparable接口,重寫compareTo方法,通過(guò)變量age實(shí)現(xiàn)自定義排序 測(cè)試方法如下:

 

  1. public static void main(String[] args) { 
  2.     Set<Person> treeSet = new TreeSet<>(); 
  3.     System.out.println("treeSet初始容量大?。?quot;+treeSet.size()); 
  4.     treeSet.add(new Person("李一",18)); 
  5.     treeSet.add(new Person("李二",17)); 
  6.     treeSet.add(new Person("李三",19)); 
  7.     treeSet.add(new Person("李四",21)); 
  8.     treeSet.add(new Person("李五",20)); 
  9.  
  10.     System.out.println("treeSet容量大小:"+treeSet.size()); 
  11.     System.out.println("按照年齡從小到大,自定義排序結(jié)果:"); 
  12.     //迭代器遍歷 
  13.     Iterator<Person> iterator = treeSet.iterator(); 
  14.     while (iterator.hasNext()){ 
  15.         Person person = iterator.next(); 
  16.         System.out.print(person.toString() + ","); 
  17.     } 

輸出結(jié)果:

 

  1. treeSet初始容量大小:0 
  2. treeSet容量大?。? 
  3. 按照年齡從小到大,自定義排序結(jié)果: 
  4. 李二:17,李一:18,李三:19,李五:20,李四:21, 

方法二

第二種方法是在TreeSet初始化階段,Person不用實(shí)現(xiàn)Comparable接口,將Comparator接口以內(nèi)部類的形式作為參數(shù),初始化進(jìn)去,方法如下:

 

  1. public static void main(String[] args) { 
  2.     //自定義排序 
  3.     Set<Person> treeSet = new TreeSet<>(new Comparator<Person>(){ 
  4.         @Override 
  5.         public int compare(Person o1, Person o2) { 
  6.             if(o1 == null || o2 == null){ 
  7.                 //不用比較 
  8.                 return 0; 
  9.             } 
  10.             //從小到大進(jìn)行排序 
  11.             return o1.getAge() - o2.getAge(); 
  12.         } 
  13.     }); 
  14.     System.out.println("treeSet初始容量大?。?quot;+treeSet.size()); 
  15.     treeSet.add(new Person("李一",18)); 
  16.     treeSet.add(new Person("李二",17)); 
  17.     treeSet.add(new Person("李三",19)); 
  18.     treeSet.add(new Person("李四",21)); 
  19.     treeSet.add(new Person("李五",20)); 
  20.  
  21.     System.out.println("treeSet容量大?。?quot;+treeSet.size()); 
  22.     System.out.println("按照年齡從小到大,自定義排序結(jié)果:"); 
  23.     //迭代器遍歷 
  24.     Iterator<Person> iterator = treeSet.iterator(); 
  25.     while (iterator.hasNext()){ 
  26.         Person person = iterator.next(); 
  27.         System.out.print(person.toString() + ","); 
  28.     } 

輸出結(jié)果:

 

  1. treeSet初始容量大?。? 
  2. treeSet容量大?。? 
  3. 按照年齡從小到大,自定義排序結(jié)果: 
  4. 李二:17,李一:18,李三:19,李五:20,李四:21, 

需要注意的是,TreeSet不能添加為空的元素,否則會(huì)報(bào)空指針錯(cuò)誤!

05. EnumSet

EnumSet 是一個(gè)與枚舉類型一起使用的專用 Set 集合,繼承自AbstractSet抽象類。與 HashSet、LinkedHashSet 、TreeSet 不同的是,EnumSet 元素必須是Enum的類型,并且所有元素都必須來(lái)自同一個(gè)枚舉類型,EnumSet 定義源碼如下:

 

  1. public abstract class EnumSet<E extends Enum<E>> extends AbstractSet<E> 
  2.     implements Cloneable, java.io.Serializable { 
  3.     ...... 

EnumSet是一個(gè)虛類,不能直接通過(guò)實(shí)例化來(lái)獲取對(duì)象,只能通過(guò)它提供的靜態(tài)方法來(lái)返回EnumSet實(shí)現(xiàn)類的實(shí)例。

EnumSet的實(shí)現(xiàn)類有兩個(gè),分別是RegularEnumSet、JumboEnumSet兩個(gè)類,兩個(gè)實(shí)現(xiàn)類都繼承自EnumSet。

EnumSet會(huì)根據(jù)枚舉類型中元素的個(gè)數(shù),來(lái)決定是返回哪一個(gè)實(shí)現(xiàn)類,當(dāng) EnumSet元素中的元素個(gè)數(shù)小于或者等于64,就會(huì)返回RegularEnumSet實(shí)例;當(dāng)EnumSet元素個(gè)數(shù)大于64,就會(huì)返回JumboEnumSet實(shí)例。

這一點(diǎn),我們可以從源碼中看出,源碼如下:

 

  1. public static <E extends Enum<E>> EnumSet<E> noneOf(Class<E> elementType) { 
  2.     Enum<?>[] universe = getUniverse(elementType); 
  3.     if (universe == null
  4.         throw new ClassCastException(elementType + " not an enum"); 
  5.     //當(dāng)元素個(gè)數(shù)小于或者等于 64 的時(shí)候,返回 RegularEnumSet 
  6.     if (universe.length <= 64) 
  7.         return new RegularEnumSet<>(elementType, universe); 
  8.     else 
  9.         //大于64,返回 JumboEnumSet 
  10.         return new JumboEnumSet<>(elementType, universe); 

noneOf是EnumSet中一個(gè)靜態(tài)方法,用于判斷是返回哪一個(gè)實(shí)現(xiàn)類。

我們來(lái)看看當(dāng)元素個(gè)數(shù)小于等于64的時(shí)候,使用RegularEnumSet的類,源碼如下:

 

  1. class RegularEnumSet<E extends Enum<E>> extends EnumSet<E> { 
  2.  
  3.     /**元素為long型*/ 
  4.     private long elements = 0L; 
  5.  
  6.     /**添加元素*/ 
  7.     public boolean add(E e) { 
  8.         typeCheck(e); 
  9.  
  10.         long oldElements = elements; 
  11.         //二進(jìn)制運(yùn)算,獲取元素 
  12.         elements |= (1L << ((Enum<?>)e).ordinal()); 
  13.         return elements != oldElements; 
  14.     } 

RegularEnumSet 通過(guò)二進(jìn)制運(yùn)算得到結(jié)果,直接使用long來(lái)存放元素。

我們?cè)賮?lái)看看當(dāng)元素個(gè)數(shù)大于64的時(shí)候,使用JumboEnumSet的類,源碼如下:

 

  1. class JumboEnumSet<E extends Enum<E>> extends EnumSet<E> { 
  2.  
  3.     /**元素為long型*/ 
  4.     private long elements = 0L; 
  5.  
  6.     /**添加元素*/ 
  7.     public boolean add(E e) { 
  8.         typeCheck(e); 
  9.  
  10.         int eOrdinal = e.ordinal(); 
  11.         int eWordNum = eOrdinal >>> 6; 
  12.  
  13.         long oldElements = elements[eWordNum]; 
  14.         //二進(jìn)制運(yùn)算 
  15.         elements[eWordNum] |= (1L << eOrdinal); 
  16.         //使用數(shù)組來(lái)操作元素 
  17.         boolean result = (elements[eWordNum] != oldElements); 
  18.         if (result) 
  19.             size++; 
  20.         return result; 
  21.     } 

JumboEnumSet 也是通過(guò)二進(jìn)制運(yùn)算得到結(jié)果,使用long來(lái)存放元素,但是它是使用數(shù)組來(lái)存放元素。

二者相比,RegularEnumSet 效率比 JumboEnumSet 高些,因?yàn)椴僮鞑襟E少,大多數(shù)情況下返回的是 RegularEnumSet,只有當(dāng)枚舉元素個(gè)數(shù)超過(guò) 64 的時(shí)候,會(huì)使用 JumboEnumSet。

添加元素

新建一個(gè)EnumEntity的枚舉類型,定義2個(gè)參數(shù)。

 

  1. public enum EnumEntity { 
  2.     WOMAN,MAN; 

創(chuàng)建一個(gè)空的 EnumSet!

 

  1. //創(chuàng)建一個(gè) EnumSet,內(nèi)容為空 
  2. EnumSet<EnumEntity> noneSet = EnumSet.noneOf(EnumEntity.class); 
  3. System.out.println(noneSet); 

輸出結(jié)果:

 

  1. [] 

創(chuàng)建一個(gè) EnumSet,并將枚舉類型的元素全部添加進(jìn)去!

 

  1. //創(chuàng)建一個(gè) EnumSet,將EnumEntity 元素內(nèi)容添加到EnumSet中 
  2. EnumSet<EnumEntity> allSet = EnumSet.allOf(EnumEntity.class); 
  3. System.out.println(allSet); 

輸出結(jié)果:

 

  1. [WOMAN, MAN] 

創(chuàng)建一個(gè) EnumSet,添加指定的枚舉元素!

 

  1. //創(chuàng)建一個(gè) EnumSet,添加 WOMAN 到 EnumSet 中 
  2. EnumSet<EnumEntity> customSet = EnumSet.of(EnumEntity.WOMAN); 
  3. System.out.println(customSet); 

查詢?cè)?/p>

EnumSet與HashSet、LinkedHashSet、TreeSet一樣,通過(guò)迭代器或者 for 循環(huán)來(lái)遍歷元素,方法如下:

 

  1. EnumSet<EnumEntity> allSet = EnumSet.allOf(EnumEntity.class); 
  2. for (EnumEntity enumEntity : allSet) { 
  3.     System.out.print(enumEntity + ","); 

輸出結(jié)果:

 

  1. WOMAN,MAN, 

06. 總結(jié)

 

 

HashSet 是一個(gè)輸入輸出無(wú)序的 Set 集合,元素不重復(fù),底層基于 HashMap 的 key 來(lái)實(shí)現(xiàn),元素可以為空,如果添加的元素為對(duì)象,對(duì)象需要重寫 equals() 和 hashCode() 方法來(lái)約束是否為相同的元素。

LinkedHashSet 是一個(gè)輸入輸出有序的 Set 集合,繼承自 HashSet,元素不重復(fù),底層基于 LinkedHashMap 的 key來(lái)實(shí)現(xiàn),元素也可以為空,LinkedHashMap 使用循環(huán)鏈表結(jié)構(gòu)來(lái)保證輸入輸出有序。

TreeSet 是一個(gè)排序的 Set 集合,元素不可重復(fù),底層基于 TreeMap 的 key來(lái)實(shí)現(xiàn),元素不可以為空,默認(rèn)按照自然排序來(lái)存放元素,也可以使用 Comparable 和 Comparator 接口來(lái)比較大小,實(shí)現(xiàn)自定義排序。

EnumSet 是一個(gè)與枚舉類型搭配使用的專用 Set 集合,在 jdk1.5 中加入。EnumSet 是一個(gè)虛類,有2個(gè)實(shí)現(xiàn)類 RegularEnumSet、JumboEnumSet,不能顯式的實(shí)例化改類,EnumSet 會(huì)動(dòng)態(tài)決定使用哪一個(gè)實(shí)現(xiàn)類,當(dāng)元素個(gè)數(shù)小于等于64的時(shí)候,使用 RegularEnumSet;大于 64的時(shí)候,使用JumboEnumSet類,EnumSet 其內(nèi)部使用位向量實(shí)現(xiàn),擁有極高的時(shí)間和空間性能,如果元素是枚舉類型,推薦使用 EnumSet。

07. 參考

1、JDK1.7&JDK1.8 源碼

 

2、程序園 - java集合-EnumMap與EnumSet

責(zé)任編輯:華軒 來(lái)源: Java極客技術(shù)
相關(guān)推薦

2019-11-11 14:51:19

Java數(shù)據(jù)結(jié)構(gòu)Properties

2011-07-04 10:39:57

Web

2013-11-14 15:53:53

AndroidAudioAudioFlinge

2021-03-16 08:54:35

AQSAbstractQueJava

2022-09-26 09:01:15

語(yǔ)言數(shù)據(jù)JavaScript

2015-08-06 14:02:31

數(shù)據(jù)分析

2018-03-15 09:13:43

MySQL存儲(chǔ)引擎

2022-12-02 09:13:28

SeataAT模式

2009-11-30 16:46:29

學(xué)習(xí)Linux

2017-07-02 18:04:53

塊加密算法AES算法

2019-01-07 15:29:07

HadoopYarn架構(gòu)調(diào)度器

2021-07-20 15:20:02

FlatBuffers阿里云Java

2012-05-21 10:06:26

FrameworkCocoa

2009-12-25 15:49:43

Linux rescu

2009-11-17 17:31:58

Oracle COMM

2021-07-19 11:54:15

MySQL優(yōu)先隊(duì)列

2023-12-04 13:22:00

JavaScript異步編程

2010-07-26 12:57:12

OPhone游戲開發(fā)

2016-10-14 13:53:05

JavascriptDOMWeb

2016-10-14 14:32:58

JavascriptDOMWeb
點(diǎn)贊
收藏

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