java源码阅读 - LinkedHashSet

29 篇文章 2 订阅
19 篇文章 0 订阅
LinkedHashSet是基于LinkedHashMap实现的Set集合,继承自HashSet,保证元素的插入顺序。文章介绍了LinkedHashSet的类声明、构造方法,以及其依赖HashMap和LinkedHashMap的知识。尽管没有提供额外的方法,LinkedHashSet利用HashSet的方法实现添加、删除和查找操作。
摘要由CSDN通过智能技术生成

往期文章

目录

一、介绍

前面文章中我们从源码详细介绍了继承于HashMap的LinkedHashMap,并通过图片示例讲解了LinkedHashMap是如何在HashMap的哈希表上将各个节点通过双向链表串起来的。

也讲解了基于HashMap实现的HashSet,那么是否存在类似于LinkedHashMap原理的一种Set集合?答案是肯定的,而且是我们本篇文章要讲的LinkedHashSet

顾名思义,LinkedHashSet是基于LinkedHashMap实现的一个Set集合。

另外,本片文章虽然不长,但是对前置知识点有着很强的依赖,需要掌握的前置知识有:HashMap(必选)红黑树(可选)LinkedHashMap(必选)HashSet(必选)

二、类的声明

public class LinkedHashSet<E> extends HashSet<E>
							    implements Set<E>, Cloneable, java.io.Serializable

从类的声明中可以看到

  • 继承HashSet,表示LinkedHashSet是对HashSet的扩展。
  • 实现set接口,满足Set集合的定义
  • 实现了Cloneable接口,提供了对象克隆方法,但请注意,是浅克隆
  • 实现了Serializable接口,支持序列化

三、构造方法

前面我们在讲HashSet的构造方法时,其中有一个构造方法我们做了特殊对待,如下所示

HashSet(int initialCapacity, float loadFactor, boolean dummy) {
    map = new LinkedHashMap<>(initialCapacity, loadFactor);
}

该构造方法创建的map对象的类型是LinkedHashMap,不同于其他构造方法创建的HashMap对象。

而且我们有个关键点不要忽略,在LinkedHashMap中,双向链表的遍历顺序通过构造方法指定,如果没有指定,则使用默认顺序为插入顺序,即accessOrder=false。因此,上面的构造方法所创建的LinkedHashMap对象的双向链表遍历顺序为插入顺序

且该构造方法就是为了给其子类LinkedHashSet使用的。我们往下看

  • 无参构造

    创建LinkedHashMap实例为内部属性,并指定底层哈希表的初始容量为16,加载因子为0.75

    public LinkedHashSet() {
        super(16, .75f, true);
    }
    
  • 指定初始容量

    创建LinkedHashMap实例为内部属性,并指定底层哈希表的初始容量为initialCapacity,加载因子为0.75

    public LinkedHashSet(int initialCapacity) {
        super(initialCapacity, .75f, true);
    }
    
  • 指定初始容量和加载因子

    创建LinkedHashMap实例为内部属性,并指定底层哈希表的初始容量为initialCapacity,加载因子为loadFactor

    public LinkedHashSet(int initialCapacity, float loadFactor) {
        super(initialCapacity, loadFactor, true);
    }
    
  • 通过集合构造

    虽然说LinkedHashSet的底层是LinkedHashMap,但终究还是哈希表+双向链表,需要对哈希表的容量进行计算以避免频繁的扩容。

    创建LinkedHashMap实例作为内部对象后,通过addAll()方法将集合中的元素逐一保存,addAll()方法作为一个批量保存模版由其父类AbstractCollection提供,其中的add()方法由父类HashSet实现,这是设计模式—模版方法的体现。

    public LinkedHashSet(Collection<? extends E> c) {
        super(Math.max(2*c.size(), 11), .75f, true);
        addAll(c);
    }
    
    public boolean addAll(Collection<? extends E> c) {
        boolean modified = false;
        for (E e : c)
            if (add(e))
                modified = true;
        return modified;
    }
    

四、最后

看了LinkedHashSet的源码后,发现它只提供了以上几个构造函数,却没有提供各个方法。这是因为它继承于HashSet,因此HashSet中提供的方法都是可以被LinkedHashSet对象调用的,如add()、remove()、contains()等方法。所以不再过多介绍,

五、结论

  • LinkedHashSet内部维护一个LinkedHashMap对象,其底层数据结构为哈希表+链表+红黑树+双向链表
  • LinkedHashSet对内部双向链表的遍历顺序为插入顺序


纸上得来终觉浅,绝知此事要躬行。

————————————————我是万万岁,我们下期再见————————————————

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

理想万岁万万岁

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值
>