LinkedHashSet学习(基于JDK1.8)

1. 类的特性

LinkedHashSet的类注释,提供了以下信息

  • LinkedHashSet基于哈希表和链表实现了Set接口
    • 允许有且只有一个null
    • 在所有的元素中维护了一个双向链表,可以维护元素的插入顺序
  • 性能:
    • 与HashSet一样,在散列均匀的情况下,基本操作(add、remove、contains)的时间复杂度为 O ( 1 ) O(1) O(1)
    • 但实际性能稍逊于HashSet,因为维护元素间的双向链表需要一定的开销。
    • LinkedHashSet元素的遍历,不再基于桶,而是基于链表,遍历时间与元素个数成正比
  • LinkedHashSet是非线程安全的,多线程访问,可以使用Collections.synchronizedSet()将其转为线程安全的set类型
  • 使用fail-fast 迭代器,一旦创建好迭代器,除非使用迭代器自身的remove方法,其他任何修改结构的方法,都将触发迭代器抛出ConcurrentModificationException 异常

总结:

  • 使用哈希表加(双向)链表的结构,允许null值,可以维护元素的插入顺序
  • 基本操作的性能为 O ( 1 ) O(1) O(1),遍历是基于链表而非桶
  • 非线程安全,使用fail-fast 迭代器

疑问:

  • 回想其余set类实现,LinkedHashSet应该是基于LinkedHashMap实现的。
  • 为何类注释中,没有说LinkedHashSet支持访问顺序呢?
  • 只是说,通过双向链表维护了元素的插入顺序

2. LinkedHashSet & LinkedHashMap

2.1 LinkedHashSet的实现如此简单

  • 查看LinkedHashSet源码,其结构如下
    在这里插入图片描述
  • 除了构造函数,没有常见的set类的关键方法,甚至没有成员变量
  • 让人感觉很神奇,为何实现如此简单?

2.2 类图

  • LinkedHashSet类的定义如下

    public class LinkedHashSet<E> extends HashSet<E>
        implements Set<E>, Cloneable, java.io.Serializable 
    
  • 类图如下
    在这里插入图片描述

  • 查看LinkedHashMap的类图,二者非常相似,简直是照葫芦画瓢

2.3 关联分析

  • LinkedHashMap基于HashMap实现,对一些关键方法进行了重写,从而在所有的entry中维护一个双向链表
  • HashSet基于HashMap实现,存在一个default构造函数,使用子类LinkedHashMap初始化HashMap
    • dummy入参:无意义的参数,只是为了实现重载,与其他的构造函数相区别
    HashSet(int initialCapacity, float loadFactor, boolean dummy) {
        map = new LinkedHashMap<>(initialCapacity, loadFactor);
    }
    
  • 从Java的多态可知,通过该构造函数初始化的map字段,实际执行时将调用子类LinkedHashMap的相关方法

巧妙之处来了:

  • LinkedHashSet的构造函数,实际都调用HashSet的上述default构造函数

  • 也就是说,LinkedHashSet中的map字段,实际为LinkedHashMap类型

  • 这样,所有entry之间就存在一个双向链表,即LinkedHashSet的所有元素之间存在一个双向链表

  • 从而,LinkedHashSet中元素是有序的,为元素的插入顺序

    // 指定初始化容量和loadFactor的空set
    public LinkedHashSet(int initialCapacity, float loadFactor) {
        super(initialCapacity, loadFactor, true);
    }
    
    // 指定初始化容量、使用默认loadFactor的空set
    public LinkedHashSet(int initialCapacity) {
        super(initialCapacity, .75f, true);
    }
    
    // 使用默认值构建一个空set
    public LinkedHashSet() {
        super(16, .75f, true);
    }
    
    // 基于指定的元素构建一个set
    public LinkedHashSet(Collection<? extends E> c) {
        super(Math.max(2*c.size(), 11), .75f, true);
        addAll(c);
    }
    

2.4 为何不支持访问顺序?

  • 从HashSet的default构造函数可以看出,构建的LinkedHashMap将默认使用插入顺序

    HashSet(int initialCapacity, float loadFactor, boolean dummy) {
        map = new LinkedHashMap<>(initialCapacity, loadFactor);
    }
    
  • 因此,基于LinkedHashMap的LinkedHashSet,也将使用插入顺序

  • 没有其他的构造函数可以提供一个具有访问顺序的LinkedHashMap,LinkedHashSet自然也不会支持访问顺序

3. 总结

关于LinkedHashSet

  • 继承HashSet类,巧妙的依靠Java的继承与多态,建立起与LinkedHashMap之间的联系
  • 实际上,基于LinkedHashMap实现了Set接口

与HashSet的区别

  • 最大的区别:元素是有序的,支持插入顺序
  • 先学习List类:ArrayList、Vector、LinkedList
  • 再学习Map类:TreeMap(先学习红黑树)、HashMap、LinkedHashMap
  • 最后学习Set类:TreeSet、HashSet、LinkedHashSet;与上述Map类一起,对照学习
### 回答1: 在JDK 1.8中,API文档是用英文编写的,但是我可以为你提供一些英文和中文对照的示例。 1. Object类 (Object class) - equals()方法: 判断两个对象是否相等。 - toString()方法: 返回对象的字符串表示。 - hashCode()方法: 返回对象的哈希码值。 2. String类 (String class) - length()方法: 返回字符串的长度。 - charAt()方法: 返回指定位置的字符。 - substring()方法: 返回子串。 3. ArrayList类 (ArrayList class) - add()方法: 将元素添加到ArrayList。 - get()方法: 获取指定位置的元素。 - remove()方法: 删除指定位置的元素。 4. HashMap类 (HashMap class) - put()方法: 将键值对添加到HashMap。 - get()方法: 根据键获取对应的值。 - remove()方法: 根据键删除键值对。 5. DateTimeFormatter类 - parse()方法: 将字符串解析为日期时间对象。 - format()方法: 将日期时间对象格式化为字符串。 请注意,这只是一小部分JDK 1.8 API中的方法示例,整个API文档提供了更多详细的说明和示例。您可以参考官方的JDK 1.8 API文档以获取更全面的信息。 ### 回答2: 1. java.lang 包下的常用类: - Object:对象类 - Class:类类 - String:字符串类 - StringBuilder:字符串构建器类 - StringBuffer:字符串缓冲类 - System:系统类,提供与系统相关的方法和属性 - Math:数学类,提供数学相关的方法和常量 - Boolean:布尔类 - Byte:字节类 - Character:字符类 - Short:短整数类 - Integer:整数类 - Long:长整数类 - Float:单精度浮点数类 - Double:双精度浮点数类 - Enum:枚举类 - Throwable:可抛出对象类,是所有错误和异常的基类 2. java.util 包下的常用类: - ArrayList:动态数组类 - LinkedList:链表类 - HashMap:哈希映射类 - LinkedHashMap:有序哈希映射类 - TreeSet:树集类 - TreeMap:树映射类 - HashSet:哈希集合类 - LinkedHashSet:有序哈希集合类 - Queue:队列接口 - PriorityQueue:优先队列类 - Stack:栈类 - Iterator:迭代器接口 - ListIterator:列表迭代器接口 - Comparator:比较器接口 - Calendar:日历类 - Date:日期类 - Timer:定时器类 3. java.io 包下的常用类: - File:文件类 - InputStream:输入流接口 - OutputStream:输出流接口 - Reader:字符输入流接口 - Writer:字符输出流接口 - BufferedReader:缓冲字符输入流类 - BufferedWriter:缓冲字符输出流类 - FileReader:文件字符输入流类 - FileWriter:文件字符输出流类 - FileInputStream:文件字节输入流类 - FileOutputStream:文件字节输出流类 - ObjectInputStream:对象输入流类 - ObjectOutputStream:对象输出流类 - PrintWriter:打印输出流类 - RandomAccessFile:随机访问文件类 - ByteArrayInputStream:字节数组输入流类 - ByteArrayOutputStream:字节数组输出流类 - CharArrayReader:字符数组输入流类 - CharArrayWriter:字符数组输出流类 这些只是jdk1.8 API中的一部分常用类和接口,还有很多其他类和接口不一一列举。 ### 回答3: JDK 1.8中的API是指Java开发工具包1.8版本中提供的所有类、接口和方法,它们用于开发Java应用程序。下面是一些常用API的中英文对照: 1. java.lang包: - Object: 对象 - String: 字符串 - Integer: 整型 - Double: 双精度浮点型 - Math: 数学计算 2. java.util包: - ArrayList: 数组列表 - LinkedList: 链表 - HashMap: 哈希表 - TreeSet: 树集合 - PriorityQueue: 优先队列 3. java.io包: - File: 文件 - InputStream: 输入流 - OutputStream: 输出流 - BufferedReader: 缓冲读取器 - BufferedWriter: 缓冲写入器 4. java.awt包: - Frame: 框架 - Button: 按钮 - Label: 标签 - TextField: 文本框 - Checkbox: 复选框 5. java.net包: - Socket: 套接字 - ServerSocket: 服务器套接字 - URL: 统一资源定位符 - HttpURLConnection: HTTP连接 这只是JDK 1.8 API中一小部分常用类和接口的中英文对照,实际上API包含了更多的类和接口,用于满足不同的编程需求。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值