JavaSE——集合6:Set接口实现类—LinkedHashSet

目录

一、LinkedHashSet的全面说明

二、LinkedHashSet源码解读

1.初始化数组长度为16,临界值为12

2.执行add方法中的add方法

3.执行put方法

4.计算hash值

5.table=null,扩容

6.创建新的结点,存放数据

7.返回null之前判断元素个数是否超过临界值

8.基本数据类型会被自动装箱

9.重复值和HashSet一样,不会被添加

10.双向链表示例

三、练习题


一、LinkedHashSet的全面说明

  1. LinkedHashSet是HashSet的子类
  2. LinkedHashSet底层是一个LinkedHashMap,底层维护了一个数组+双向链表
  3. LinkedHashSet根据元素的hashCode值来决定元素的存储位置 ,同时使用链表维护元素的次序,这使得元素看起来是以插入顺序保存的
  4. LInkedHashSet不允许添加重复元素
  5. 只允许一个null值

二、LinkedHashSet源码解读

  1. 在LinkedHashSet中维护了一个hash表和双向链表(LinkedHashSet有head和tail)
  2. 每一个节点有before和after属性,这样可以形成双向链表
  3. 在添加一个元素时,先求hash值,再求索引,确定该元素在table的位置,然后将添加的元素加入到双向链表(如果已经存在,则不添加[原则和hashSet一样])
  4. 这样的话,遍历LinkedHashSet也能确保,插入顺序和遍历顺序一致
java 复制代码
public class LinkedHashSetSource {
    public static void main(String[] args) {
        // LinkedHashSet的底层机制
        Set set = new LinkedHashSet();
        set.add(new String("AA"));
        set.add(456);
        set.add(456);
        set.add(new Customer("张三", 1001));
        set.add(123);
        set.add("hello");

        System.out.println("set=" + set);
        /*
           // 继承关系是在内部类完成
           // HashMap.Node  说明Node是一个静态方法
           // static class Node<K,V> implements Map.Entry<K,V>

           static class Entry<K,V> extends HashMap.Node<K,V> {
               Entry<K,V> before, after;
               Entry(int hash, K key, V value, Node<K,V> next) {
                   super(hash, key, value, next);
               }
           }*/
    }
}
  1. LinkedHashSet 加入顺序和取出元素/数据的顺序一致;

  2. LinkedHashSet 底层维护的是一个LinkedHashMap(是HashMap的子类);

  3. LinkedHashSet 底层结构 (数组table+双向链表);

  4. 第一次添加时,直接将 数组table 扩容到 16 ,临界值是12,存放的结点类型是 LinkedHashMap$Entry;

  5. 数组是 HashMapNode\[\] 存放的元素/数据是 LinkedHashMapEntry类型。

1.初始化数组长度为16,临界值为12

LinkedHashSet继承了HashSet

2.执行add方法中的add方法

3.执行put方法

4.计算hash值

5.table=null,扩容

6.创建新的结点,存放数据

7.返回null之前判断元素个数是否超过临界值

8.基本数据类型会被自动装箱

9.重复值和HashSet一样,不会被添加

10.双向链表示例

三、练习题

Car类(属性:name,price),如果name和price一样,则认为是相同元素,就不能添加。

代码实现:

java 复制代码
public class LinkedHashSetExercise {
    public static void main(String[] args) {
        LinkedHashSet linkedHashSet = new LinkedHashSet();
        linkedHashSet.add(new Car("奥拓", 1000)); // OK
        linkedHashSet.add(new Car("奥迪", 300000)); // OK
        linkedHashSet.add(new Car("法拉利", 1000000)); // OK
        linkedHashSet.add(new Car("奥迪", 300000)); // 加入不了
        linkedHashSet.add(new Car("保时捷", 7000000)); // OK
        linkedHashSet.add(new Car("奥迪", 300000)); // 加入不了

        for (Object o : linkedHashSet) {
            System.out.println(o);
        }
        // Car{name='奥拓', price=1000.0}
        // Car{name='奥迪', price=300000.0}
        // Car{name='法拉利', price=1000000.0}
        // Car{name='保时捷', price=7000000.0}
    }
}

class Car {
    private String name;
    private double price;

    public Car(String name, double price) {
        this.name = name;
        this.price = price;
    }

    // 注意:必须同时重写equals()和hashCode()方法
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Car car = (Car) o;
        return Double.compare(car.price, price) == 0 && Objects.equals(name, car.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, price);
    }

    @Override
    public String toString() {
        return "Car{" +
                "name='" + name + '\'' +
                ", price=" + price +
                '}';
    }
}
相关推荐
风象南11 分钟前
SpringBoot的5种日志输出规范策略
java·spring boot·后端
咖啡啡不加糖17 分钟前
深入理解MySQL死锁:从原理、案例到解决方案
java·数据库·mysql
zimoyin19 分钟前
Compose Multiplatform 实现自定义的系统托盘,解决托盘乱码问题
java
啾啾Fun36 分钟前
【Java微服务组件】分布式协调P4-一文打通Redisson:从API实战到分布式锁核心源码剖析
java·redis·分布式·微服务·lua·redisson
消失的旧时光-19431 小时前
Android USB 通信开发
android·java
惊鸿一博1 小时前
java_网络服务相关_gateway_nacos_feign区别联系
java·开发语言·gateway
朝新_7 小时前
【多线程初阶】阻塞队列 & 生产者消费者模型
java·开发语言·javaee
立莹Sir7 小时前
Calendar类日期设置进位问题
java·开发语言
季鸢9 小时前
Java设计模式之状态模式详解
java·设计模式·状态模式
@yanyu6669 小时前
springboot实现查询学生
java·spring boot·后端