zoukankan      html  css  js  c++  java
  • java源码 -- LinkedHashSet

    问题

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

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

    (3)LinkedHashSet是有序的吗?

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

    源码分析

    LinkedHashSet继承自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来初始化HashSet中的map。

    现在这个逻辑应该很清晰了,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是不支持按访问顺序对元素排序的,只能按插入顺序排序。

    参考:https://www.cnblogs.com/tong-yuan/p/LinkedHashSet.html

  • 相关阅读:
    antdesign vue 官网地址
    WMS智能仓储管理系统-干货-产品介绍
    C# 判断文件夹与文件是否存在
    C# 获取文件大小,创建时间,文件信息,FileInfo类的属性表
    问题2:采购发票、材料出库单、到货单,打开时提示某某插件是否安装! 问题
    问题一:u813.0销售出库单-EAI接口报-在对应所需名称或序数的集合中,未找到项目
    SqlServer数据库sa登录失败,错误18456
    解决<compilation debug="true" targetFramework="4.0"> 问题
    SQL Server 2005无法远程连接的解决方法
    Windows Server 2008 IIS 7.5 HTTP 错误 404.3
  • 原文地址:https://www.cnblogs.com/FondWang/p/11938869.html
Copyright © 2011-2022 走看看