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 +
                '}';
    }
}
相关推荐
何苏三月6 分钟前
设计模式 - 单例模式(懒汉式、饿汉式、静态内部类、枚举)
java·单例模式
Renas_TJOvO10 分钟前
排序算法汇总
java·数据结构·算法
秋恬意20 分钟前
Java 反射机制详解
java·开发语言
爱上语文28 分钟前
LeetCode每日一题
java·算法·leetcode
ღ᭄ꦿ࿐Never say never꧂36 分钟前
重生之我在Java世界------学工厂设计模式
java·设计模式·简单工厂模式·应用场景
尘浮生1 小时前
Java项目实战II基于Spring Boot的火锅店管理系统设计与实现(开发文档+数据库+源码)
java·开发语言·数据库·spring boot·后端·微信小程序·旅游
wrx繁星点点1 小时前
桥接模式:解耦抽象与实现的利器
android·java·开发语言·jvm·spring cloud·intellij-idea·桥接模式
羊小猪~~1 小时前
C/C++语言基础--C++模板与元编程系列二类模板、全特化、偏特化、编译模型简介、实现简单Vetctor等…………)
java·c语言·开发语言·c++·visual studio code·visual studio
l138494274511 小时前
C语言储存变量 java循环语句和循环跳转
java·c语言·开发语言·算法
HHppGo2 小时前
java_封装
java·开发语言