[学习笔记-Java集合-10] Set - LinkedHashSet源码分析

介绍

上一节咱们说HashSet中的元素是无序的,那么有没有什么办法保证Set中的元素是有序的呢?java

答案是固然能够。 LinkedHashSet就有这个功能,它是怎么实现有序的呢?多线程

源码分析

LinkedHashSet继承自HashSet,让咱们直接上源码来看看它们有什么不一样。ide

package java.util;

// LinkedHashSet继承自HashSet
public class LinkedHashSet<E>
    extends HashSet<E>
    implements Set<E>, Cloneable, java.io.Serializable {

    private static final long serialVersionUID = -2851667679971038690L;

    // 传入容量和装载因子
    public LinkedHashSet(int initialCapacity, float loadFactor) {
        super(initialCapacity, loadFactor, true);
    }

    // 只传入容量, 装载因子默认为0.75
    public LinkedHashSet(int initialCapacity) {
        super(initialCapacity, .75f, true);
    }

    // 使用默认容量16, 默认装载因子0.75
    public LinkedHashSet() {
        super(16, .75f, true);
    }

    // 将集合c中的全部元素添加到LinkedHashSet中
    // 好奇怪, 这里计算容量的方式又变了
    // HashSet中使用的是Math.max((int) (c.size()/.75f) + 1, 16)
    // 这一点有点不得其解, 是做者偷懒?
    public LinkedHashSet(Collection<? extends E> c) {
        super(Math.max(2*c.size(), 11), .75f, true);
        addAll(c);
    }

    // 可分割的迭代器, 主要用于多线程并行迭代处理时使用
    @Override
    public Spliterator<E> spliterator() {
        return Spliterators.spliterator(this, Spliterator.DISTINCT | Spliterator.ORDERED);
    }
}

完了,这是所有源码了。源码分析

能够看到,LinkedHashSet中一共提供了5个方法,其中4个是构造方法,还有一个是迭代器。学习

4个构造方法都是调用父类的super(initialCapacity, loadFactor, true);这个方法。this

这个方法就是咱们上一节说过,HashSet的一个不是public的构造方法。线程

// HashSet的构造方法
    HashSet(int initialCapacity, float loadFactor, boolean dummy) {
        map = new LinkedHashMap<>(initialCapacity, loadFactor);
    }

如上所示,这个构造方法里面使用了LinkedHashMap来初始化HashSet中的map。code

如今这个逻辑应该很清晰了,LinkedHashSet继承自HashSet,它的添加、删除、查询等方法都是直接用的HashSet的,惟一的不一样就是它使用LinkedHashMap存储元素。排序

总结

(1)LinkedHashSet的底层使用LinkedHashMap存储元素。继承

(2)LinkedHashSet是有序的,它是按照插入的顺序排序的。

思考

经过上面的学习,咱们知道LinkedHashSet底层使用LinkedHashMap存储元素,而LinkedHashMap是支持按元素访问顺序遍历元素的,也就是能够用来实现LRU的。

那么,LinkedHashSet支持按元素访问顺序排序吗?

让咱们一块儿来分析下。

首先,LinkedHashSet全部的构造方法都是调用HashSet的同一个构造方法,以下:

// HashSet的构造方法
    HashSet(int initialCapacity, float loadFactor, boolean dummy) {
        map = new LinkedHashMap<>(initialCapacity, loadFactor);
    }
    
## 而后,经过调用LinkedHashMap的构造方法初始化map,以下所示: 

    public LinkedHashMap(int initialCapacity, float loadFactor) {
        super(initialCapacity, loadFactor);
        accessOrder = false;
    }

能够看到,这里把accessOrder写死为false了。

因此,LinkedHashSet是不支持按访问顺序对元素排序的,只能按插入顺序排序。

相关文章
相关标签/搜索