精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

深入淺出的分析 Set集合

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

 01. 摘要

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

[[282307]]

言歸正傳,廢話咱們也不多說了,相信使用過 Set 集合類的朋友都知道,Set集合的特點主要有:元素不重復、存儲無序的特點。

啥意思呢?你可以理解為,向一個瓶子里面扔東西,這些東西沒有記號是第幾個放進去的,但是有一點就是這個瓶子里面不會有重樣的東西。

細細思考,你會發現, Set 集合的這些特性正處于 List 集合和 Map 集合之間,為什么這么說呢?之前的集合文章中,咱們了解到,List 集合的特點就是存取有序,本質是一個有序數組,每個元素依次按照順序存儲;Map 集合主要用于存放鍵值對,雖然底層也是用數組存放,但是元素在數組中的下標是通過哈希算法計算出來的,數組下標無序。

而 Set 集合,在元素存儲方面,注重獨立無二的特性,如果某個元素在集合中已經存在,不會存儲重復的元素,同時,集合存儲的是元素,不像 Map 集合那樣存儲的是鍵值對。

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

 

 

 

 

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

  • HashSet 是一個輸入輸出無序的集合,集合中的元素基于 HashMap 的 key 實現,元素不可重復;
  • LinkedHashSet 是一個輸入輸出有序的集合,集合中的元素基于 LinkedHashMap 的 key 實現,元素也不可重復;
  • TreeSet 是一個排序的集合,集合中的元素基于 TreeMap 的 key 實現,同樣元素不可重復;
  • EnumSet 是一個與枚舉類型一起使用的專用 Set 集合,其中 RegularEnumSet 和 JumboEnumSet 不能單獨實例化,只能由 EnumSet 來生成,同樣元素不可重復;

下面咱們來對各個主要實現類進行一一分析!

02. HashSet

HashSet 是一個輸入輸出無序的集合,底層基于 HashMap 來實現,HashSet 利用 HashMap 中的key元素來存放元素,這一點我們可以從源碼上看出來,閱讀源碼如下:

 

  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.         //默認實例化一個 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,是一個非空對象,源碼部分如下:

 

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

可以分析出,當進行add()的時候,等價于

 

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

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

從源碼分析上可以看出,HashSet 正是使用了 HashMap 的這一特性,實現存儲元素下標無序、元素不會重復的特點。

remove方法

HashSet 的刪除方法,同樣如此,也是基于 HashMap 的底層實現,源碼如下:

 

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

查詢方法

HashSet 沒有像 List、Map 那樣提供 get 方法,而是使用迭代器或者 for 循環來遍歷元素,方法如下:

 

  1. public static void main(String[] args) { 
  2.     Set<String> hashSet = new HashSet<String>(); 
  3.     System.out.println("HashSet初始容量大小:"+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.     //相同元素會自動覆蓋 
  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.     //增強for循環 
  22.     for (String str : hashSet) { 
  23.         System.out.print(str + ","); 
  24.     } 

輸出結果:

 

  1. HashSet初始容量大小:0 
  2. HashSet容量大小:4 
  3. null,1,2,3, 
  4. =========== 
  5. null,1,2,3, 

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

03. LinkedHashSet

LinkedHashSet 是一個輸入輸出有序的集合,繼承自 HashSet,但是底層基于 LinkedHashMap 來實現。

如果你之前了解過 LinkedHashMap,那么你一定知道,它也繼承自 HashMap,唯一有區別的是,LinkedHashMap 底層數據結構基于循環鏈表實現,并且數組指定了頭部和尾部,雖然數組的下標存儲無序,但是卻可以通過數組的頭部和尾部,加上循環鏈表,依次可以查詢到元素存儲的過程,從而做到輸入輸出有序的特點。

如果還不了解 LinkedHashMap 的實現過程,可以參閱集合系列中關于 LinkedHashMap 的實現過程文章。

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

 

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

查詢源碼,super調用的方法,源碼如下:

 

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

add方法

LinkedHashSet沒有重寫add方法,而是直接調用HashSet的add()方法,因為map的實現類是LinkedHashMap,所以此處是向LinkedHashMap中添加元素,當進行add()的時候,等價于

 

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

remove方法

LinkedHashSet也沒有重寫remove方法,而是直接調用HashSet的刪除方法,因為LinkedHashMap沒有重寫remove方法,所以調用的也是HashMap的remove方法,源碼如下:

 

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

查詢方法

同樣的,LinkedHashSet 沒有提供 get 方法,使用迭代器或者 for 循環來遍歷元素,方法如下:

 

  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容量大小:"+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.     //增強for循環 
  22.     for (String str : linkedHashSet) { 
  23.         System.out.print(str + ","); 
  24.     } 

輸出結果:

 

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

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

04. TreeSet

TreeSet 是一個排序的集合,實現了NavigableSet、SortedSet、Set接口,底層基于 TreeMap 來實現。TreeSet 利用 TreeMap 中的key元素來存放元素,這一點我們也可以從源碼上看出來,閱讀源碼,類定義如下:

 

  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.     /**對象初始化*/ 
  8.     public TreeSet() { 
  9.         //默認實例化一個 TreeMap 對象 
  10.         this(new TreeMap<E,Object>()); 
  11.     } 
  12.      
  13.     //對象初始化調用的方法 
  14.     TreeSet(NavigableMap<E,Object> m) { 
  15.         this.m = m; 
  16.     } 

new TreeSet<>()對象實例化的時候,表達的意思,可以簡化為如下:

 

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

因為TreeMap實現了NavigableMap接口,所以沒啥問題。

 

  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,也是是一個非空對象,源碼部分如下:

 

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

可以分析出,當進行add()的時候,等價于

 

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

TreeMap 類主要功能在于,給添加的集合元素,按照一個的規則進行了排序,默認以自然順序進行排序,當然也可以自定義排序,比如測試方法如下:

 

  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.     //默認自然排序,key為升序 
  8.     System.out.println("默認 排序結果:" + initMap.toString()); 
  9.     //自定義排序,在TreeMap初始化階段傳入Comparator 內部對象 
  10.     Map comparatorMap = new TreeMap<String, String>(new Comparator<String>() { 
  11.         @Override 
  12.         public int compare(String o1, String o2){ 
  13.             //根據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("自定義 排序結果:" + comparatorMap.toString()); 

輸出結果:

 

  1. 默認 排序結果:{1=a, 2=b, 3=c, 4=d} 
  2. 自定義 排序結果:{4=d, 3=c, 2=b, 1=a} 

相信使用過TreeMap的朋友,一定知道TreeMap會自動將key按照一定規則進行排序,TreeSet正是使用了TreeMap這種特性,來實現添加的元素集合,在輸出的時候,其結果是已經排序好的。

如果您沒看過源碼TreeMap的實現過程,可以參閱集合系列文章中TreeMap的實現過程介紹,或者閱讀 jdk 源碼。

remove方法

TreeSet 的刪除方法,同樣如此,也是基于 TreeMap 的底層實現,源碼如下:

 

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

查詢方法

TreeSet 沒有重寫 get 方法,而是使用迭代器或者 for 循環來遍歷元素,方法如下:

 

  1. public static void main(String[] args) { 
  2.     Set<String> treeSet = new TreeSet<>(); 
  3.     System.out.println("treeSet初始容量大小:"+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容量大小:"+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.     //增強for循環 
  20.     for (String str : treeSet) { 
  21.         System.out.print(str + ","); 
  22.     } 

輸出結果:

 

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

自定義排序

使用自定義排序,有 2 種方法,第一種在需要添加的元素類,實現Comparable接口,重寫compareTo方法來實現對元素進行比較,實現自定義排序。

方法一

 

  1. /** 
  2.   * 創建實體類Person實現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.     } 

創建一個Person實體類,實現Comparable接口,重寫compareTo方法,通過變量age實現自定義排序 測試方法如下:

 

  1. public static void main(String[] args) { 
  2.     Set<Person> treeSet = new TreeSet<>(); 
  3.     System.out.println("treeSet初始容量大小:"+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("按照年齡從小到大,自定義排序結果:"); 
  12.     //迭代器遍歷 
  13.     Iterator<Person> iterator = treeSet.iterator(); 
  14.     while (iterator.hasNext()){ 
  15.         Person person = iterator.next(); 
  16.         System.out.print(person.toString() + ","); 
  17.     } 

輸出結果:

 

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

方法二

第二種方法是在TreeSet初始化階段,Person不用實現Comparable接口,將Comparator接口以內部類的形式作為參數,初始化進去,方法如下:

 

  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.             //從小到大進行排序 
  11.             return o1.getAge() - o2.getAge(); 
  12.         } 
  13.     }); 
  14.     System.out.println("treeSet初始容量大小:"+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容量大小:"+treeSet.size()); 
  22.     System.out.println("按照年齡從小到大,自定義排序結果:"); 
  23.     //迭代器遍歷 
  24.     Iterator<Person> iterator = treeSet.iterator(); 
  25.     while (iterator.hasNext()){ 
  26.         Person person = iterator.next(); 
  27.         System.out.print(person.toString() + ","); 
  28.     } 

輸出結果:

 

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

需要注意的是,TreeSet不能添加為空的元素,否則會報空指針錯誤!

05. EnumSet

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

 

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

EnumSet是一個虛類,不能直接通過實例化來獲取對象,只能通過它提供的靜態方法來返回EnumSet實現類的實例。

EnumSet的實現類有兩個,分別是RegularEnumSet、JumboEnumSet兩個類,兩個實現類都繼承自EnumSet。

EnumSet會根據枚舉類型中元素的個數,來決定是返回哪一個實現類,當 EnumSet元素中的元素個數小于或者等于64,就會返回RegularEnumSet實例;當EnumSet元素個數大于64,就會返回JumboEnumSet實例。

這一點,我們可以從源碼中看出,源碼如下:

 

  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.     //當元素個數小于或者等于 64 的時候,返回 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中一個靜態方法,用于判斷是返回哪一個實現類。

我們來看看當元素個數小于等于64的時候,使用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.         //二進制運算,獲取元素 
  12.         elements |= (1L << ((Enum<?>)e).ordinal()); 
  13.         return elements != oldElements; 
  14.     } 

RegularEnumSet 通過二進制運算得到結果,直接使用long來存放元素。

我們再來看看當元素個數大于64的時候,使用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.         //二進制運算 
  15.         elements[eWordNum] |= (1L << eOrdinal); 
  16.         //使用數組來操作元素 
  17.         boolean result = (elements[eWordNum] != oldElements); 
  18.         if (result) 
  19.             size++; 
  20.         return result; 
  21.     } 

JumboEnumSet 也是通過二進制運算得到結果,使用long來存放元素,但是它是使用數組來存放元素。

二者相比,RegularEnumSet 效率比 JumboEnumSet 高些,因為操作步驟少,大多數情況下返回的是 RegularEnumSet,只有當枚舉元素個數超過 64 的時候,會使用 JumboEnumSet。

添加元素

新建一個EnumEntity的枚舉類型,定義2個參數。

 

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

創建一個空的 EnumSet!

 

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

輸出結果:

 

  1. [] 

創建一個 EnumSet,并將枚舉類型的元素全部添加進去!

 

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

輸出結果:

 

  1. [WOMAN, MAN] 

創建一個 EnumSet,添加指定的枚舉元素!

 

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

查詢元素

EnumSet與HashSet、LinkedHashSet、TreeSet一樣,通過迭代器或者 for 循環來遍歷元素,方法如下:

 

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

輸出結果:

 

  1. WOMAN,MAN, 

06. 總結

 

 

HashSet 是一個輸入輸出無序的 Set 集合,元素不重復,底層基于 HashMap 的 key 來實現,元素可以為空,如果添加的元素為對象,對象需要重寫 equals() 和 hashCode() 方法來約束是否為相同的元素。

LinkedHashSet 是一個輸入輸出有序的 Set 集合,繼承自 HashSet,元素不重復,底層基于 LinkedHashMap 的 key來實現,元素也可以為空,LinkedHashMap 使用循環鏈表結構來保證輸入輸出有序。

TreeSet 是一個排序的 Set 集合,元素不可重復,底層基于 TreeMap 的 key來實現,元素不可以為空,默認按照自然排序來存放元素,也可以使用 Comparable 和 Comparator 接口來比較大小,實現自定義排序。

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

07. 參考

1、JDK1.7&JDK1.8 源碼

 

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

責任編輯:華軒 來源: Java極客技術
相關推薦

2019-11-11 14:51:19

Java數據結構Properties

2021-03-16 08:54:35

AQSAbstractQueJava

2011-07-04 10:39:57

Web

2013-11-14 15:53:53

AndroidAudioAudioFlinge

2009-11-30 16:46:29

學習Linux

2022-12-02 09:13:28

SeataAT模式

2021-07-20 15:20:02

FlatBuffers阿里云Java

2017-07-02 18:04:53

塊加密算法AES算法

2019-01-07 15:29:07

HadoopYarn架構調度器

2012-05-21 10:06:26

FrameworkCocoa

2022-09-26 09:01:15

語言數據JavaScript

2018-03-15 09:13:43

MySQL存儲引擎

2015-08-06 14:02:31

數據分析

2023-03-20 09:48:23

ReactJSX

2009-12-25 15:49:43

Linux rescu

2018-11-09 16:24:25

物聯網云計算云系統

2021-04-27 08:54:43

ConcurrentH數據結構JDK8

2022-11-09 08:06:15

GreatSQLMGR模式

2012-02-21 13:55:45

JavaScript

2022-10-31 09:00:24

Promise數組參數
點贊
收藏

51CTO技術棧公眾號

国产精品二区一区二区aⅴ| 精品久久久久久无码国产| 国产高清免费观看| 亚洲一区一卡| 播播国产欧美激情| 日本天堂在线播放| 欧美日韩尤物久久| 亚洲午夜激情av| 亚洲国产日韩美| 秋霞网一区二区| 麻豆国产精品官网| 欧美在线视频免费| 男女免费视频网站| 91综合在线| 亚洲人成电影网站色…| 成人一区二区三区仙踪林| 99久久精品一区二区成人| 亚洲成av人片在www色猫咪| 中文字幕一区二区三区乱码| 日本高清中文字幕二区在线| 成人激情黄色小说| 亚洲精品欧美日韩| 一级片视频免费| 日日夜夜免费精品| 欧美综合在线观看| www.99re7.com| 久久久9色精品国产一区二区三区| 亚洲欧美日韩一区二区在线 | 中文字幕av一区二区| 超碰男人的天堂| 成人在线tv视频| 日韩欧美美女一区二区三区| 久久久福利影院| 9999精品免费视频| 在线观看一区不卡| 国产激情在线观看视频| 一个人www视频在线免费观看| 亚洲一区av在线| 久久久久久久久久伊人| a视频在线播放| 亚洲卡通欧美制服中文| 国产成人精品免费看在线播放| 在线观看h片| 国产精品系列在线| 日韩欧美第二区在线观看| 欧美男男同志| 久久亚洲一级片| 欧美日韩在线观看一区| 女人偷人在线视频| 久久影院电视剧免费观看| 久久久久久国产精品mv| 香蕉久久国产av一区二区| 99国产欧美另类久久久精品| 精品欧美一区二区三区久久久| 网站黄在线观看| 91免费小视频| 日本精品国语自产拍在线观看| 免费在线国产| 国产精品三级电影| 中文字幕欧美日韩一区二区| а√天堂资源地址在线下载| 亚洲激情五月婷婷| 三上悠亚久久精品| av日韩亚洲| 欧美日韩亚洲丝袜制服| 亚洲18在线看污www麻豆| 亚洲日本va午夜在线电影| 亚洲精品一区二区三区蜜桃下载 | 国产精品色婷婷久久58| 性做爰过程免费播放| 大桥未久在线播放| 91豆麻精品91久久久久久| 婷婷免费在线观看| 日本在线视频一区二区三区| 亚洲黄在线观看| 第一次破处视频| 亚洲天天影视网| 97人人爽人人喊人人模波多| 成人三级视频| 欧美影院一区二区三区| www.色欧美| 超碰在线成人| 一本一本久久a久久精品牛牛影视| 一级性生活免费视频| 国内揄拍国内精品久久| 国产成人一区二区| www.色呦呦| 91香蕉视频污在线| 超碰免费在线公开| 高端美女服务在线视频播放| 欧美在线观看禁18| 日本五十肥熟交尾| 久久视频精品| 午夜精品www| 91精品国产乱码久久久久| 波多野结衣精品在线| 亚洲狠狠婷婷综合久久久| 欧美14一18处毛片| 欧美性高清videossexo| avtt中文字幕| 99精品在线| 青青草国产精品一区二区| 国产精品怡红院| 国产偷国产偷亚洲高清人白洁| 97精品国产97久久久久久粉红| 亚洲精品动漫| 亚洲精品在线观看视频| 2014亚洲天堂| 久热精品在线| 精品国产一区二区三区麻豆免费观看完整版 | 久久免费大视频| 欧美一级片在线播放| 亚洲va欧美va| 中文字幕永久在线不卡| 日本国产在线播放| 亚洲精品在线播放| 久久影视电视剧免费网站清宫辞电视| 午夜婷婷在线观看| 波多野结衣亚洲一区| 国产在线无码精品| 亚洲欧洲一二区| 一区二区欧美在线| www.久久久久久久| 91社区在线播放| 99热亚洲精品| 国产精品成人自拍| 色综合五月天导航| 国产av精国产传媒| 一区在线中文字幕| 亚洲精品自拍网| 精品国产乱码久久久久久1区2匹| 欧洲精品在线视频| 天堂√在线中文官网在线| 午夜欧美大尺度福利影院在线看| 色男人天堂av| 欧美精品首页| 国产成人免费电影| √最新版天堂资源网在线| 日韩精品一区二区在线| 欧美成欧美va| 成人性色生活片| 妞干网在线视频观看| www.豆豆成人网.com| 色黄久久久久久| 艳妇乳肉豪妇荡乳av| 亚洲国产成人午夜在线一区| 欧美成人黄色网址| 日韩激情图片| 国产欧美日韩精品专区| 在线视频91p| 欧美一卡2卡3卡4卡| 中文字幕av播放| 国产成人av电影| 水蜜桃色314在线观看| 日韩极品少妇| 国产成人高清激情视频在线观看| 都市激情在线视频| 欧美日韩极品在线观看一区| 99久久99久久精品国产| 成人自拍视频在线| 777久久久精品一区二区三区| 五月综合久久| 国产噜噜噜噜噜久久久久久久久 | 五月天久久久| 99九九电视剧免费观看| 鲁鲁在线中文| 一区二区三区视频免费| 国产免费福利视频| 五月婷婷欧美视频| 日韩福利在线视频| 国产精品1区2区3区在线观看| 欧美精品一区二区三区三州| 性感美女一区二区在线观看| 欧美一级久久久久久久大片| 丰满少妇高潮久久三区| 成人午夜视频免费看| 国产精品秘入口18禁麻豆免会员| 欧美猛男男男激情videos| 国产美女精彩久久| 国产黄色大片在线观看| 日韩精品小视频| 中文字幕你懂的| 亚洲激情男女视频| www.超碰97| 久久99精品一区二区三区| av片在线免费| 国产不卡av一区二区| 成人免费看吃奶视频网站| 波多野结衣乳巨码无在线观看| 国产一区二区三区在线视频| av免费观看网址| 日本精品一区二区三区高清| 成人免费视频网站入口::| 久久亚洲捆绑美女| 人偷久久久久久久偷女厕| 精品国产一级片| 欧美午夜激情视频| 欧美a级片免费看| 99久久精品免费看国产免费软件| 国产av天堂无码一区二区三区| av中文一区| 成人动漫视频在线观看完整版| 欧美福利在线播放| 欧美成人精品h版在线观看| 日本不卡免费播放| 欧美麻豆精品久久久久久| 亚洲男人的天堂在线视频| 亚洲免费在线播放| 蜜臀久久99精品久久久久久| 97aⅴ精品视频一二三区| 激情久久综合网| 日本女优在线视频一区二区| 日韩精品在线中文字幕| 久久蜜桃av| 日韩精品福利视频| 婷婷五月色综合香五月| 91精品入口蜜桃| 深夜日韩欧美| 国产精品久久久久久久久久小说| 涩涩涩视频在线观看| 久久久久久国产精品| 国产成人l区| 日韩在线中文字| 触手亚洲一区二区三区| 日韩黄色在线免费观看| 天堂av中文字幕| 亚洲成人网av| 免费看黄网站在线观看| 日韩欧美中文一区| 99精品人妻无码专区在线视频区| 欧美日韩高清一区| 在线播放成人av| 欧美日韩免费视频| 亚洲一级av毛片| 欧美日韩黄色一区二区| 一级特黄录像免费看| 欧美区在线观看| 91麻豆国产视频| 欧美精品少妇一区二区三区| 亚洲综合一区中| 7777精品伊人久久久大香线蕉最新版| 岳乳丰满一区二区三区| 欧美日韩亚洲丝袜制服| 国产精品久久久久久久久毛片 | 亚洲综合婷婷久久| 久久99国内精品| 三级av免费看| 高清不卡在线观看av| 久久久久99人妻一区二区三区| 国产v综合v亚洲欧| 李丽珍裸体午夜理伦片| 99麻豆久久久国产精品免费 | 人妻无码中文字幕免费视频蜜桃| 欧美videofree性高清杂交| 粉嫩av一区二区夜夜嗨| 日韩精品视频在线免费观看| 黄网在线观看| 中文字幕欧美日韩| 超碰在线免费播放| 久久久这里只有精品视频| 欧美中文字幕在线观看视频| 亚洲国产99| 日本成年人网址| 日韩av网站在线观看| gogogo高清免费观看在线视频| 国产综合色精品一区二区三区| 亚洲精品无码久久久久久久| va亚洲va日韩不卡在线观看| 亚洲一区视频在线播放| 国产精品国产精品国产专区不蜜 | 成人在线视频成人| 两个人的视频www国产精品| 国模私拍视频在线播放| 欧美有码在线视频| 欧美男女视频| 国产精品美女黄网| blacked蜜桃精品一区| 9191国产视频| 男女av一区三区二区色多| 中文字幕第22页| 91麻豆高清视频| 久久人妻无码aⅴ毛片a片app| 亚洲成人激情av| 超碰在线免费97| 欧美va亚洲va在线观看蝴蝶网| 日本福利片高清在线观看| 久久天天躁狠狠躁夜夜躁2014| 啊啊啊久久久| 91精品视频在线看| 宅男在线一区| 欧妇女乱妇女乱视频| 日韩精品久久理论片| 蜜臀aⅴ国产精品久久久国产老师| 国产午夜精品在线观看| 成人观看免费视频| 欧美午夜性色大片在线观看| a视频免费在线观看| 正在播放欧美一区| 不卡视频观看| 91青草视频久久| 精品国产一区二区三区香蕉沈先生 | 国产又粗又猛又爽视频| 亚洲国产精品一区二区www| 在线视频欧美亚洲| 亚洲美女在线观看| 九色91在线| 亚洲一区精品电影| 青草国产精品| 日本wwww视频| 东方aⅴ免费观看久久av| 国产一区二区精彩视频| 在线免费亚洲电影| 亚洲欧美日韩免费| 欧美激情中文网| 久久av偷拍| 一区二区三区欧美成人| 久久久亚洲人| 人妻无码中文久久久久专区| 亚洲一区二区三区视频在线播放| 国产精品毛片一区二区在线看舒淇| 亚洲视频欧美视频| 亚洲精品一区| 麻豆传媒一区| 国产精品久久国产愉拍| 亚洲国产精品无码久久久久高潮| 一区二区在线观看免费 | 精品香蕉在线观看视频一| 青青青草视频在线| **亚洲第一综合导航网站| 91嫩草亚洲精品| 日本中文字幕精品—区二区| 国产日韩欧美精品综合| 免费观看日批视频| 国产午夜一区二区| 朝桐光一区二区| 日本一区二区三区免费观看| 日韩精品乱码免费| 成人激情五月天| 欧美日韩在线播| 一本一道波多野毛片中文在线| 国产精品视频资源| 欧美大黑bbbbbbbbb在线| 国产精品v日韩精品v在线观看| 国产精品久久久久一区二区三区| 中文字幕乱码中文字幕| 最近更新的2019中文字幕| 中文成人在线| 日韩欧美一级在线| 成人高清av在线| 国产欧美一区二区三区在线看蜜臂| 亚洲欧美激情视频| 欧美xx视频| 亚洲精品视频一区二区三区| 久久精品99久久久| 国产精品成人免费观看| 精品国产乱码久久久久久闺蜜| 色老头在线一区二区三区| 欧洲av一区| 精品一区二区三区影院在线午夜| 印度午夜性春猛xxx交| 亚洲成人网在线观看| 一区二区三区短视频| 日韩免费av电影| 狠狠色丁香婷综合久久| 精品无码免费视频| 亚洲老司机av| 亚洲综合视频| 欧美 日韩 亚洲 一区| 欧美激情一区在线观看| 国产亲伦免费视频播放| 欧美亚洲免费电影| 色狮一区二区三区四区视频| 中文字幕在线播放一区二区| 欧美日韩国内自拍| 在线视频自拍| 国模精品娜娜一二三区| 另类专区欧美蜜桃臀第一页| 久久久久成人精品无码| 亚洲老板91色精品久久| av在线精品| 黄色片视频在线免费观看| 国产精品久久久久毛片软件| 黑人乱码一区二区三区av| 国产精品高潮呻吟久久av无限| 欧美黄色免费| 免费看黄色三级| 欧美精品一区二区久久婷婷| 国产精品伦一区二区| 国产极品尤物在线| 亚洲欧洲www| 蜜桃视频在线观看网站| 国产精品久久精品国产| 麻豆专区一区二区三区四区五区| 国产午夜在线播放| 久久精品中文字幕电影| 伊甸园亚洲一区| 亚洲区 欧美区| 欧美精品色一区二区三区| 中文一区一区三区高中清不卡免费 | 成人资源www网在线最新版|