Java集合框架知识整理

Java集合框架主要由CollectionMap两个根接口及其子接口、实现类组成。

集合类关系梳理

  • Collection接口继承了Iterable接口,依赖了PredicateSpliteratorStream接口(这些均为Java8新增),Iterable接口依赖了Iterator接口。
    • List接口继承自Collection接口,依赖了UnaryOperator接口(Java8新增)、ListIteratorComparator接口
    • Queue接口继承了Collection接口
      • Deque接口继承了Queue接口
      • BlockingQueue接口继承了Queue接口
        • BlockingDeque接口继承了BlockingQueueDeque接口
        • TransferQueue接口继承了BlockingQueue接口
    • AbstractCollection抽象类实现了Collection接口,
      • AbstractList抽象类继承了AbstractCollection抽象类,实现了List接口,依赖了ListRandomAccessCloneableSerializable接口
        • ArrayList类继承了AbstractList抽象类,实现了ListRandomAccessCloneableSerializable接口
        • AbstractSequentialList抽象类继承了AbstractList抽象类
          • LinkedList类继承了AbstractSequentialList抽象类,实现了ListDequeCloneableSerializable接口
        • CopyOnWriteArrayList实现了ListRandomAccessCloneableSerializable接口
        • Vector类继承了AbstractList抽象类,实现了ListRandomAccessCloneableSerializable接口
          • Stack类继承了Vector
      • AbstractQueue抽象类继承了AbstractCollection接口,实现了Queue接口
        • SynchronousQueue类继承了AbstractQueue接口,实现了BlockingQueueSerializable接口,依赖了CollectionSpliterator接口
        • ArrayBlockingQueue类继承了AbstractQueue接口,实现了BlockingQueueSerializable接口
        • LinkedBlockingQueue类继承了AbstractQueue接口,实现了BlockingQueueSerializable接口
        • PriorityBlockingQueue类继承了AbstractQueue接口,实现了BlockingQueueSerializable接口,聚合了Comparator接口,依赖了CollectionComparatorComparable接口
        • DelayQueue类继承了AbstractQueue接口,实现了BlockingQueue接口
        • LinkedBlockingDeque类继承了AbstractQueue接口,实现了BlockingDequeSerializable接口
        • PriorityQueue类继承了AbstractQueue接口
        • LinkedTransferQueue类继承了AbstractQueue接口,实现了TransferQueueSerializable接口(Java7新增)
      • ConcurrentLinkedDeque类继承了AbstractCollection抽象类,实现了DequeCloneableSerializable接口
      • ArrayDeque类继承了AbstractCollection抽象类,实现了DequeSerializable接口
    • Set接口继承自Collection接口
      • AbstractSet抽象类继承了AbstractCollection抽象类,实现了Set接口
        • HashSet类继承了AbstractSet抽象类,实现了SetCloneableSerializable接口,聚合了HashMap
          • LinkedHashSet类继承了HashSet类,实现了SetCloneableSerializable接口
        • TreeSet类继承了AbstractSet抽象类,实现了NavigableSetCloneableSerializable接口,聚合了NavigableMap,依赖了ComparatorSortedSet接口
        • EnumSet抽象类继承了AbstractSet抽象类,实现了CloneableSerializable接口,依赖了ComparatorSortedSet接口
          • RegularEnumSet类继承了EnumSet抽象类
          • JumboEnumSet类继承了EnumSet抽象类
        • ConcurrentSkipListSet类继承了AbstractSet抽象类,实现了NavigableSetCloneableSerializable接口
        • CopyOnWriteArraySet类继承了AbstractSet抽象类,实现了Serializable接口,聚合了CopyOnWriteArrayList类,依赖了PredicateConsumer接口
      • SortedSet接口继承自Set接口,依赖了Comparator接口
        • NavigableSet接口继承自SortedSet接口(Java6新增)
  • Map接口依赖了SetCollectionBiConsumerFunctionBiFunction接口,Map.Entry是Map中的内部接口
    • AbstractMap抽象类实现了Map接口,聚合了CollectionSet接口
      • HashMap类继承了AbstractMap抽象类,实现了MapCloneableSerializable接口,依赖了CollectionSet接口
        • LinkedHashMap继承了HashMap类,实现了Map接口,依赖了CollectionSetConsumerBiConsumer接口
      • TreeMap类继承了AbstractMap抽象类,实现了NavigableMapCloneableSerializable接口,依赖了ComparatorSortedMapCollectionSetBiConsumerBiFunction接口
      • EnumMap类继承了AbstractMap抽象类,实现了CloneableSerializable接口,依赖了AbstractSet类,CollectionSet接口
      • WeakHashMap类继承了AbstractMap抽象类,实现了Map接口,依赖了CollectionSetConsumerBiConsumerBiFunction接口
      • IdentityHashMap类继承了AbstractMap抽象类,实现了MapSerializableCloneable接口,依赖了CollectionSetConsumerBiConsumerBiFunction接口
      • ConcurrentHashMap类继承了AbstractMap抽象类,实现了ConcurrentMapSerializable接口,依赖了ComparableParameterizedTypeCollectionSetSpliteratorConsumerBiConsumerFunctionBiFunctionToDoubleFunctionDoubleBinaryOperator等接口
      • ConcurrentSkipListMap类继承了AbstractMap抽象类,实现了ConcurrentNavigableMapCloneableSerializable接口,聚合了Comparator接口,依赖了CollectionSetConsumerBiConsumerBiFunctionNavigableSet接口
    • SortedMap接口继承自Map接口,依赖了SetCollectionComparator接口
      • NavigableMap接口继承了SortedMap接口,依赖了NavigableSet接口
      • ConcurrentNavigableMap接口继承了ConcurrentMapNavigableMap接口,聚合了NavigableSet接口
    • ConcurrentMap接口继承了Map接口,依赖了BiConsumerBiFunction接口
    • Hashtable类继承了Dictionary抽象类,实现了MapCloneableSerializable接口,聚合了CollectionSet接口,依赖了EnumerationBiConsumerBiFunction接口
      • Properties类继承了Hashtable
  • CollectionsCollection的辅助工具类,依赖了上述大多数接口和类
  • Arrays是数组的辅助工具类,依赖了上述一些接口和类

集合UML关系图

Java Collection UML类关系图如下:

Java Collection UML类关系图

Java Map UML类关系图如下:

Java Map UML类关系图

各集合接口、类的介绍

  • CollectionCollection是最基本集合接口,它定义了一组允许重复的对象。Collection接口派生了三个子接口ListSetQueueCollection所有实现类的遍历都可以使用Iterator接口或者是foreach来循环。
    • ListList代表有序、可重复的集合。
      • ArrayList:底层使用数组的形式来实现,排列有序可重复,查询速度快、增删数据慢,线程不安全,效率高。ArrayList创建时的大小为0;当加入第一个元素时,进行第一次扩容时,默认容量大小为10,每次扩容都以当前数组大小的1.5倍去扩容。
      • Vector:底层使用数组的形式来实现,排列有序可重复,查询速度快、增删数据慢,线程安全,效率低。Vector创建时的默认大小为10Vector每次扩容都以当前数组大小的2倍去扩容。当指定了capacityIncrement之后,每次扩容仅在原先基础上增加capacityIncrement个单位空间。ArrayListVectoraddgetsize方法的复杂度都为O(1)remove方法的复杂度为O(n)
        • StackVector的一个子类,是标准的先进后出(FILO, First In Last Out)的栈。底层通过数组实现的,线程安全。
      • LinkedList:底层使用双向循环链表的数据结构来实现,排列有序可重复,查询速度慢、增删数据快,线程不安全。
      • CopyOnWriteArrayList:底层使用Copy-On-Write的优化策略实现,适用于读多写少的场景,同ArrayList功能相似,线程安全。CopyOnWriteArrayList在某些情况下比Collections.synchronizedList(List list)有更好的性能。缺点是:内存占用大和数据一致性问题,只能保证最终一致性。
    • SetSet代表无序、不可重复的集合。
      • HastSet:底层使用Hash表来实现,内部使用了HashMap,排列无序不可重复,存取速度快,线程不安全。
        • LinkedHashSet:底层采用Hash表存储,并用双向链表记录插入顺序,排列有序不可重复,存取速度较HashSet略慢,比TreeSet快,线程不安全。
      • TreeSet:底层使用红黑树来实现,内部使用了NavigableMap,按自然顺序或者自定义顺序存放、不可重复,线程不安全。
      • CopyOnWriteArraySet:底层使用Copy-On-Write的优化策略实现,适用于读多写少的场景,内部使用了CopyOnWriteArrayList,同HastSet功能相似,线程安全。
      • ConcurrentSkipListSet:底层使用跳跃列表来实现,适用于高并发的场景,内部使用了ConcurrentNavigableMap,同TreeSet功能相似,线程安全。
      • EnumSet:是抽象类,只能用来存储Enum常量或其子类,不能存储其它类型,EnumSet有两种实现方式,RegularEnumSetJumboEnumSet,但是这两种实现方式是包私有的,不能在包外访问,因此必须使用工厂方法来创建并返回EnumSet实例,不能通过构造函数来创建。EnumSet中提供了多种创建EnumSet实例的静态工厂方法,例如of方法(进行了函数重载),copyOf方法,noneOf方法等。存储效率快,线程不安全。存储枚举常量时使用EnumSet而不要用HashSet
    • QueueQueue是Java 5之后增加的集合体系,表示队列集合的相关实现,大多遵循先进先出(FIFO, First-In-First-Out)的模式。
      • PriorityQueue:即优先队列,底层基于优先堆的一个无界队列来实现,无界但可选容量界限。这个优先队列中的元素可以默认自然排序或者通过提供的Comparator(比较器)在队列实例化的时排序,而不是先进先出。不允许空值、不支持non-comparable(不可比较)的对象,每次从队列中取出的是具有最高优先权的元素,线程不安全。
      • ArrayBlockingQueue:底层基于定长数组的阻塞队列实现,即是线程安全的有界阻塞队列。ArrayBlockingQueue内部通过互斥锁保护竞争资源,实现了多线程对竞争资源的互斥访问。队列中的锁是没有分离的,所以在添加的同时就不能读取,读取的同时就不能添加,所以锁方面性能不如LinkedBlockingQueue
      • LinkedBlockingQueue:即链接队列,底层基于单向链表的阻塞队列实现,无界但可选容量界限,线程安全。队列中的锁是分离的,即添加用的是putLock,获取是takeLock,所以在添加获取方面理论上性能会高于ArrayBlockingQueue。所以LinkedBlockingQueue更适合实现生产者-消费者队列。
      • PriorityBlockingQueue:即优先阻塞队列,底层基于优先堆的一个无界队列来实现,无界但可选容量界限的阻塞队列,线程安全,功能同PriorityQueueLinkedBlockQueue相似。其所含对象的排序不是先进先出,而是依据对象的自然排序顺序或者是构造函数的Comparator决定的顺序。
      • SynchronousQueue:即同步队列,是一种线程安全无缓冲的无界阻塞队列。其操作必须是放和取交替完成的,即每个put必须等待一个take,反之亦然。
      • DelayQueue:即延迟队列,是一种有序无界阻塞队列,只有在延迟期满时才能从中提取元素,线程安全。
      • ArrayDeque:底层采用了循环数组的方式来完成双端队列的实现,无限扩展且可选容量。Java已不推荐使用Stack,而是推荐使用更高效的ArrayDeque来实现栈的功能,非线程安全。
      • LinkedBlockingDeque:底层采用了双向链表实现的双端阻塞并发队列,无限扩展且可选容量。该阻塞队列同时支持FIFOFILO两种操作方式,即可以从队列的头和尾同时操作(插入/删除),且线程安全。
      • ConcurrentLinkedDeque:底层采用了双向链表实现的双端非阻塞并发队列,无限扩展且可选容量。该队列同时支持FIFOFILO两种操作方式,即可以从队列的头和尾同时操作(插入/删除),且线程安全。
      • LinkedTransferQueue:底层采用了单向链表实现的无界传输阻塞队列,先进先出,无限扩展且可选容量线程安全。
  • MapMap代表具有映射关系的集合。
    • HashMap:底层是用链表数组Java8后又加了红黑树来实现,键无序不可重复可为null、值可重复可为null,存取速度快,线程不安全。
      • LinkedHashMap:底层是用链表数组存储,并用双向链表记录插入顺序,键有序不可重复可为null、值可重复可为null,存取速度快较HashMap略慢,比TreeMap快,线程不安全。
    • HashTable:底层是用链表数组,键无序不可重复可为null、值可重复可为null,存取速度较HashMap慢,线程安全。
      • Properties:是HashTable的子类,是<String,String>的映射,比HashTable多了loadstore两个方法,线程安全。
    • TreeMap:底层使用红黑树来实现,内部使用了Comparator,按自然顺序或自定义顺序存放键,键不可重复不可为null、值可重复可为null,存取速度较HashMap慢,线程不安全。
    • EnumMap:底层使用数组来实现,是专门为枚举类型量身定做的Map,性能更好。只能接收同一枚举类型的实例作为键值,并且由于枚举类型实例的数量相对固定并且有限,所以EnumMap使用数组来存放与枚举类型对应的值,线程不安全。
    • WeakHashMap:同HashMap基本相似。区别在于,HashMapkey保留对象的强引用,这意味着只要该HashMap对象不被销毁,该HashMap对象所有key所引用的对象不会被垃圾回收,HashMap也不会自动删除这些key所对应的key-value对象;但WeakHashMapkey只保留对实际对象的弱引用,这意味着当垃圾回收了该key所对应的实际对象后,WeakHashMap会自动删除该key对应的key-value对象。
    • IdentityHashMap:同HashMap基本相似。区别在于,在处理两个key相等时,对于普通HashMap而言,只要key1key2通过equals比较返回true时就认为key相同;在IdentityHashMap中,当且仅当两个key严格相等时(key1 = key2)时才认为两个key相同。
    • ConcurrentHashMap:底层使用锁分段技术来实现线程安全,首先将数据分成一段一段的存储,然后给每一段数据配一把锁,当一个线程占用锁访问其中一个段数据的时候,其他段的数据也能被其他线程访问。ConcurrentHashMap是由Segment数组结构和HashEntry数组结构组成。Segment是一种可重入锁ReentrantLock,在ConcurrentHashMap里扮演锁的角色,HashEntry则用于存储键值对数据。一个ConcurrentHashMap里包含一个Segment数组,Segment的结构和HashMap类似,是一种数组和链表结构, 一个Segment里包含一个HashEntry数组,每个HashEntry是一个链表结构的元素, 每个Segment守护者一个HashEntry数组里的元素,当对HashEntry数组的数据进行修改时,必须首先获得它对应的Segment锁。
    • ConcurrentSkipListMap:底层使用跳跃列表来实现,适用于高并发的场景,内部使用了ConcurrentNavigableMap,同TreeMap功能相似,是一个并发的、可排序的Map,线程安全。因此它可以在多线程环境中弥补ConcurrentHashMap不支持排序的问题。

Java集合框架功能介绍思维导图如下:

Java集合框架功能介绍思维导图

一些概念解释

  • 跳表:是一种采用了用空间换时间思想的数据结构。它会随机地将一些节点提升到更高的层次,以创建一种逐层的数据结构,以提高操作的速度。
  • 阻塞队列和非阻塞的区别:如果队列里面已经放满了,如果是阻塞队列那么线程会一直阻塞,而非阻塞对垒则会抛出异常。

一些数据结构的优缺点

  • Hash表:插入、查找最快,为O(1);如使用链表实现则可实现无锁;数据有序化需要显式的排序操作。
  • 红黑树:插入、查找为O(logn),但常数项较小;无锁实现的复杂性很高,一般需要加锁;数据天然有序。
  • SkipList:插入、查找为O(logn),但常数项比红黑树要大;底层结构为链表,可无锁实现;数据天然有序。

一些接口的主要方法梳理

Collection接口中的抽象方法

  • int size(),返回集合的大小
  • boolean isEmpty(),返回集合是否为空的布尔值
  • boolean contains(Object o),返回集合是否包含元素o的布尔值
  • Iterator<E> iterator(),返回该集合中元素的迭代器,继承自Iterable接口
  • Object[] toArray(),返回一个包含此集合中所有元素的数组
  • <T> T[] toArray(T[] a)toArray()方法的泛型版本,返回一个包含此集合中所有元素的数组,返回类型由传入数组参数的类型决定
  • boolean add(E e),返回向集合中插入元素e是否成功的布尔值
  • boolean remove(Object o),返回从集合中删除元素o是否成功的布尔值
  • boolean containsAll(Collection<?> c),返回本集合中是否完全包含集合c的布尔值,即判断集合c是否是本集合子集
  • boolean addAll(Collection<? extends E> c),将集合c中的所有元素添加到本集合中并返回
  • boolean removeAll(Collection<?> c),移除本集合中所有包含集合c的所有元素
  • default boolean removeIf(Predicate<? super E> filter),Java8新增的接口默认方法。将会批量删除符合filter条件的所有元素,该方法需要一个Predicate对象作为作为参数,Predicate也是函数式接口,因此可使用Lambda表达式作为参数。
  • boolean retainAll(Collection<?> c),返回本集合和集合c中相同的元素并存到本集合中,集合c保持不变,返回值表示的是本集合是否发生过改变。即该方法是用来求两个集合的交集,交集的结果存到本集合中,如果本集合没发生变化则返回true
  • void clear(),清空本集合中的所有元素
  • boolean equals(Object o),返回本集合是否和对象o相等的布尔值
  • int hashCode(),返回此集合的Hash码值
  • default Spliterator<E> spliterator(),在集合中创建Spliterator对象
    • Spliterator是Java 8引入的新接口,顾名思义,Spliterator可以理解IteratorSplit版本(但用途要丰富很多)。使用Iterator的时候,我们可以顺序地遍历容器中的元素,使用Spliterator的时候,我们可以将元素分割成多份,分别交于不于的线程去遍历,以提高效率。使用Spliterator每次可以处理某个元素集合中的一个元素 — 不是从Spliterator中获取元素,而是使用tryAdvance()forEachRemaining()方法对元素应用操作。但Spliterator还可以用于估计其中保存的元素数量,而且还可以像细胞分裂一样变为一分为二。这些新增加的能力让流并行处理代码可以很方便地将工作分布到多个可用线程上完成。
  • default Stream<E> stream(),返回一个顺序的Stream对象。Java8引入了Stream以实现对集合更方便地进行函数式编程。
  • default Stream<E> parallelStream(),返回一个可能并行的Stream对象。Java8新增的方法。流可以是顺序的也可以是并行的。顺序流的操作是在单线程上执行的,而并行流的操作是在多线程上并发执行的。

List接口中的额外抽象方法

  • boolean addAll(int index, Collection<? extends E> c),将指定集合c中的所有元素插入到指定索引位置处
  • default void replaceAll(UnaryOperator<E> operator),Java8新增的使用Lambda的方式,通过应用UnaryOperator获得的结果来替换列表中的每个元素
  • default void sort(Comparator<? super E> c),在比较器的基础上将本列表排序
  • E get(int index),获取本集合中指定索引位置处的元素
  • E set(int index, E element),设置或替换本集合中指定索引位置处的元素
  • void add(int index, E element),在本集合中的指定索引位置处插入指定的元素
  • E remove(int index),移除本集合中指定索引位置处的元素
  • int indexOf(Object o),返回指定元素第一次出现的索引位置
  • int lastIndexOf(Object o),返回指定元素最后出现的索引位置
  • ListIterator<E> listIterator(),返回本集合中的ListIterator迭代器
  • ListIterator<E> listIterator(int index),返回本集合中从指定索引位置开始的ListIterator迭代器
  • List<E> subList(int fromIndex, int toIndex),返回指定开始和结束索引位置的子集合

Set接口中的额外抽象方法

Map接口中的抽象方法

  • boolean containsKey,判断本Map集合中是否包含指定的key键
  • boolean containsValue,判断本Map集合中是否包含指定的value值
  • V get(Object key),根据key获取本Map集合中的value值
  • V get(Object key),向本Map集合中存放key键和value值,返回value值
  • V remove(Object key),根据key删除本Map集合中的key和value值,并返回删除的value值
  • void putAll(Map<? extends K, ? extends V> m),将指定的Map集合添加到本的Map集合当中
  • Set<K> keySet(),获取本Map集合中的所有key值,并以Set接口的结果作为返回
  • Collection<V> values(),获取本Map集合中的所有value值,并以Collection接口的结果作为返回
  • Set<Map.Entry<K, V>> entrySet(),获取本Map集合中的所有key和value值,并以Set<Map.Entry<K, V>>的结果作为返回
  • default V getOrDefault(Object key, V defaultValue),根据key获取本Map集合中的value值,如果没找到对应的值或者value值是null,则返回defaultValue的值
  • default void forEach(BiConsumer<? super K, ? super V> action),Java8新增的使用Lambda的方式遍历操作Map中的元素的默认接口方法
  • default void replaceAll(BiFunction<? super K, ? super V, ? extends V> function),Java8新增的使用Lambda的方式遍历替换Map中的元素的默认接口方法
  • default V putIfAbsent(K key, V value),Java8新增的不用写是否为null值的检测语句向Map中保存key和value的元素的默认接口方法,即如果通过key获取到的value是空的,则在调用put(key, value)方法并返回value值
  • default boolean remove(Object key, Object value),Java8新增的默认接口方法,删除给定key所对应的元素,如果value不存在、为null或者与参数中的value不等,则不能删除。即删除操作需要满足给定的值需要和map中的值相等的条件
  • default boolean replace(K key, V oldValue, V newValue),Java8新增的默认接口方法,替换给定key所对应的元素,如果value不存在、为null或者与参数中的oldValue不等,则不能替换。即替换操作需要满足给定的值需要和map中的值相等的条件
  • default V replace(K key, V value),Java8新增的默认接口方法,替换给定key所对应的元素,如果value不为null,则value值与参数中的value值做替换。
  • default V computeIfAbsent(K key, Function<? super K, ? extends V> mappingFunction),Java8新增的默认接口方法,根据key获取到的value如果不为null,则直接返回value值,否则将Lambda表达式中的结果值存放到Map中
  • default V computeIfPresent(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction),Java8新增的默认接口方法,根据key获取到的value和新计算的值如果不为null,则直接新计算的值,否则移除该key,且返回null
  • default V compute(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction),Java8新增的默认接口方法,将Lambda表达式中的结果值存放到Map中,如果计算的新值为null则返回null,且移除以前有的key和value值
  • default V merge(K key, V value, BiFunction<? super V, ? super V, ? extends V> remappingFunction),Java8新增的默认接口方法,将新计算的值覆盖Map中原key对应的value值

SortedSet接口中的额外抽象方法

  • Comparator<? super E> comparator(),返回本SortedSet集合中的Comparator比较器
  • SortedSet<E> subSet(E fromElement, E toElement),获取开始元素和结束元素之间的子SortedSet集合
  • SortedSet<E> headSet(E toElement),获取开始元素和toElement元素之间的子SortedSet集合
  • SortedSet<E> tailSet(E fromElement),获取fromElement元素和结束元素之间的子SortedSet集合
  • E first(),获取本SortedSet集合中的第一个元素
  • E last(),获取本SortedSet集合中的最后一个元素

SortedMap接口中的额外抽象方法

  • Comparator<? super K> comparator(),返回本SortedMap集合中的Comparator比较器
  • SortedMap<K,V> subMap(K fromKey, K toKey),获取开始key和结束key之间的子SortedMap集合
  • SortedMap<K,V> headMap(K toKey),获取开始key和toKey元素之间的子SortedMap集合
  • SortedMap<K,V> tailMap(K fromKey),获取fromKey元素和结束key之间的子SortedMap集合
  • K firstKey(),获取本SortedMap集合中的第一个key
  • K lastKey(),获取本SortedMap集合中的最后一个key
  • Set<K> keySet(),获取本SortedMap集合中所有key的Set集合
  • Collection<V> values(),获取本SortedMap集合中所有value的Collection集合
  • Set<Map.Entry<K, V>> entrySet(),获取本SortedMap集合中所有key和value的Map集合

Reprint please specify: Blog4Jun Java集合框架知识整理

Previous
你需要知道的Java枚举知识 你需要知道的Java枚举知识
概述定义枚举(enum全称为enumeration)类型是Java 5新增的类型,存放在java.lang包中,允许用常量来表示特定的数据片断,而且全部都以类型安全的形式来表示。 定义格式 创建枚举类型要使用enum关键字,隐含了所创建的类
2018-10-27
Next
UML各类关系介绍 UML各类关系介绍
继承关系(generalization)继承关系用一条带空心箭头的直线表示。如下图所示(A继承自B): 继承指的是一个类(称为子类、子接口)继承另外的一个类(称为父类、父接口)的功能,并可以增加它自己的新功能的能力。继承关系为is-a的关
2018-10-22