浅谈Java的LinkedHashSet源码

 更新时间:2023年09月05日 10:50:59   作者:小成同学_  
这篇文章主要介绍了浅谈Java的LinkedHashSet源码,底层是链表实现的,是set集合中唯一一个能保证怎么存就怎么取的集合对象
因为是HashSet的子类,所以也是保证元素唯一的,与HashSet的原理一样,需要的朋友可以参考下

LinkedHashSet

问题

(1)LinkedHashSet 的底层使用什么存储元素?

(2)LinkedHashSet 与 HashSet 有什么不同?

(3)LinkedHashSet 是有序的吗?

(4)LinkedHashSet 支持按元素访问顺序排序吗?

LinkedHashSet的概述和使用

A:LinkedHashSet的特点

LinkedHashSet

底层是链表实现的,是set集合中唯一一个能保证怎么存就怎么取的集合对象

因为是HashSet的子类,所以也是保证元素唯一的,与HashSet的原理一样

B:案例演示

LinkedHashSet的特点

可以保证怎么存就怎么取

package com.heima.set;
import java.util.LinkedHashSet;
public class Demo02_LinkedHashSet {
	public static void main(String[] args) {
		LinkedHashSet<String> lhs = new LinkedHashSet<>();
		lhs.add("a");
		lhs.add("a");
		lhs.add("a");
		lhs.add("a");
		lhs.add("b");
		lhs.add("c");
		lhs.add("d");
		System.out.println(lhs);
	}
}

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

答案是当然可以。

我们今天的主角 LinkedHashSet 就有这个功能,它是怎么实现有序的呢?让我们来一起学习吧。

如果你已看过前面关于 HashSet 和 HashMap,一定能够想到本文将要讲解的 LinkedHashSet 和 LinkedHashMap 其实也是一回事,前者仅仅是对后者做了一层包装,也就是说 LinkedHashSet 里面有一个 LinkedHashMap(适配器模式)

LinkedHashSet 的本质就是 LinkedHashMap。

继承体系

image-20221203211217024

源码解析

LinkedHashSet 继承了 HashSet,其增删改查等方法使用的都是 HashSet 的方法,我们直接看它的全部源码。

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); 这个方法。

这个方法长什么样呢?

还记得我们上一节说过一个不是 public 的构造方法吗?就是它。

    // HashSet的构造方法
    HashSet(int initialCapacity, float loadFactor, boolean dummy) {
        map = new LinkedHashMap<>(initialCapacity, loadFactor);
    }
					   ||
				  	   \/
	// 上面的构造器调用的是LinkedHashMap的这个构造器,其accessOrder是固定的为false,即不能实现LRU,链表只能按照元素的添加顺序进行排序。
    public LinkedHashMap(int initialCapacity, float loadFactor) {
        // 再调用HashMap的构造器
        super(initialCapacity, loadFactor);
        // accessOrder固定为false,不能为其赋值,即不支持LRU。
        accessOrder = false;
    }

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

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

上一篇我们学习了 LinkedHashMap,此时我们会想到一个问题,既然 LinkedHashSet 是基于 LinkedHashMap 实现的,那它可以实现 LRU 吗?

答案是 不可以

因为我们看到,LinkedHashSet 所有的构造方法都是调用 HashSet 的同一个构造方法,而 HashSet 中初始化 map 的构造方法,默认把 accessOrder 设置为 false 了。

所以,LinkedHashSet 是不支持按访问顺序对元素排序的,无法实现 LRU,只能按插入顺序排序。

那么,开篇那几个问题是否能回答了呢?

总结

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

(2)LinkedHashSet 是有序的,它是按照插入的顺序排序的 (不支持 LRU)

到此这篇关于浅谈Java的LinkedHashSet源码的文章就介绍到这了,更多相关Java的LinkedHashSet内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • spring mvc 读取xml文件数据库配置参数的方法

    spring mvc 读取xml文件数据库配置参数的方法

    下面小编就为大家带来一篇spring mvc 读取xml文件数据库配置参数的方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-10-10
  • java 中Comparable与Comparator详解与比较

    java 中Comparable与Comparator详解与比较

    这篇文章主要介绍了java 中Comparable与Comparator详解与比较的相关资料,需要的朋友可以参考下
    2017-04-04
  • 在Intellij Idea中使用jstl标签库的方法

    在Intellij Idea中使用jstl标签库的方法

    这篇文章主要介绍了在Intellij Idea中使用jstl标签库的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-05-05
  • JAVA如何获取客户端IP地址和MAC地址

    JAVA如何获取客户端IP地址和MAC地址

    本篇文章主要介绍了JAVA如何获取客户端IP地址和MAC地址非常具有实用价值,这里整理了详细的代码,需要的朋友可以参考下
    2017-08-08
  • Springboot实现动态定时任务流程详解

    Springboot实现动态定时任务流程详解

    通过重写SchedulingConfigurer方法实现对定时任务的操作,单次执行、停止、启动三个主要的基本功能,动态的从数据库中获取配置的定时任务cron信息,通过反射的方式灵活定位到具体的类与方法中
    2022-09-09
  • java socket大数据传输丢失问题及解决

    java socket大数据传输丢失问题及解决

    这篇文章主要介绍了java socket大数据传输丢失问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-08-08
  • 使用Scala生成随机数的方法示例

    使用Scala生成随机数的方法示例

    这篇文章主要介绍了使用Scala生成随机数的方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-06-06
  • Java匿名类和匿名函数的概念和写法

    Java匿名类和匿名函数的概念和写法

    匿名函数写法和匿名类写法的前提必须基于函数式接口匿名函数写法和匿名类写法其本质是同一个东西,只是简化写法不同使用Lambda表达式简写匿名函数时,可以同时省略实现类名、函数名,这篇文章主要介绍了Java匿名类和匿名函数的概念和写法,需要的朋友可以参考下
    2023-06-06
  • 详解Spring Boot Profiles 配置和使用

    详解Spring Boot Profiles 配置和使用

    本篇文章主要介绍了详解Spring Boot Profiles 配置和使用,具有一定的参考价值,有兴趣的可以了解一下
    2017-06-06
  • Java上传文件到服务器端的方法

    Java上传文件到服务器端的方法

    这篇文章主要为大家详细介绍了Java上传文件到服务器端的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01

最新评论