Java LinkedHashSet集合的底层原理和TreeSet集合

 更新时间:2024年10月22日 12:05:39   作者:CCH2024  
LinkedHashSet保证元素有序且唯一,底层通过双链表实现,TreeSet元素不重复且可排序,底层使用红黑树实现排序,自定义类型排序可通过实现Comparable接口或提供Comparator来定义排序规则,适用于需要大量元素快速检索的场景

Collection集合体系:

LinkedHashSet:有序、不重复、无索引。

 看一下之前的代码:

package cn.chang.d4_collection_set;
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.Set;
import java.util.TreeSet;
/**
 * 目标:整体了解下Set集合的特点
 */
public class SetTest1 {
    public static void main(String[] args) {
        // 1. 创建一个Set集合的对象
        // 无序、不重复、无索引
        //Set<Integer> set = new HashSet<Integer>();  // 多态 HashSet是一个实现类 经典代码
        Set<Integer> set = new LinkedHashSet<>();   // 有序 不重复 无索引
        //Set<Integer> set = new TreeSet<>();           // 排序升序  不重复  无索引
        set.add(666);
        set.add(555);
        set.add(555);      // 只显示一个555
        set.add(888);
        set.add(777);
        System.out.println(set);   // 不支持索引操作数据   // 无序一次
    }
}

运行结果:

可见:LinkedHashSet是无序、不重复和无索引的。

LinkedHashSet底层原理:

依然是基于哈希表(数组、链表和红黑树)实现的。

但是,它的每个元素都额外的多了一个双链表的机制记录它前后元素的位置。

通过双链表去找数据,所以就是有序的。

缺点:每个节点更占内存。

TreeSet:

特点:不重复、无索引、可排序(默认升序排序、按照元素的大小、由小到大排序)

底层是基于红黑树实现的排序。

package cn.chang.d4_collection_set;
import java.util.HashSet;
import java.util.Set;
import java.util.TreeSet;
/**
 * 目标: 掌握TreeSet集合的使用
 */
public class SetTest4 {
    public static void main(String[] args) {
        Set<Integer> set = new TreeSet<>();
        set.add(6);
        set.add(5);
        set.add(5);
        set.add(7);
        System.out.println(set);
    }
}

运行结果:

左根右,中序遍历。

注意:

对于数值类型:Integer、Double,默认是按照数值本身的大小进行升序 排序。

对于字符串类型,默认按照首字符的编号升序排列。

对于自定义类型,如Student对象,TreeSet默认是无法直接排序的。

会有如下报错。

自定义排序规则:

TreeSet集合存储自定义类型的对象时,必须指定排序规则,支持如下两种方式来指定比较规则:

方式一:

让自定义类(如学生类)实现Comparable接口,重写里面的compareTo方法来指定比较规则。

方式二:

通过调用TreeSet集合的有参构造器,可以设置Comparator对象(比较器对象,用于指定比较规则)。

方式一:

代码:

package cn.chang.d4_collection_set;
import java.util.HashSet;
import java.util.Set;
import java.util.TreeSet;
/**
 * 目标: 掌握TreeSet集合的使用
 */
public class SetTest4 {
    public static void main(String[] args) {
        Set<Student> students = new TreeSet<>();
        students.add(new Student("蜘蛛精", 23, 169.7));
        students.add(new Student("紫霞", 22, 169.8));
        students.add(new Student("至尊宝", 26, 165.5));
        students.add(new Student("牛魔王", 22, 183.5));
        System.out.println(students);
    }
}

我们可以看到牛魔王没有了。如果大小规则的相等,就重复了,牛魔王就不存了。

方式二:

package cn.chang.d4_collection_set;
import java.util.Comparator;
import java.util.HashSet;
import java.util.Set;
import java.util.TreeSet;
/**
 * 目标: 掌握TreeSet集合的使用
 */
public class SetTest4 {
    public static void main(String[] args) {
        Set<Student> students = new TreeSet<>(new Comparator<Student>() {
            @Override
            public int compare(Student s1, Student s2) {
                // 需要按照身高升序排序
                return Double.compare(s1.getHeight(), s2.getHeight());
            }
        });
        students.add(new Student("蜘蛛精", 23, 169.7));
        students.add(new Student("紫霞", 22, 169.8));
        students.add(new Student("至尊宝", 26, 165.5));
        students.add(new Student("牛魔王", 22, 183.5));
        System.out.println(students);
    }
}

运行结果:

现在我们有两台比较方案,优先按就近原则采用方法二。

代码的简化:

package cn.chang.d4_collection_set;
import java.util.Comparator;
import java.util.HashSet;
import java.util.Set;
import java.util.TreeSet;
/**
 * 目标: 掌握TreeSet集合的使用
 */
public class SetTest4 {
    public static void main(String[] args) {
//        Set<Student> students = new TreeSet<>(new Comparator<Student>() {
//            @Override
//            public int compare(Student s1, Student s2) {
//                // 需要按照身高升序排序
//                return Double.compare(s1.getHeight(), s2.getHeight());
//            }
//        });
        Set<Student> students = new TreeSet<>((o1, o2) -> Double.compare(o1.getHeight(), o2.getHeight()));
        students.add(new Student("蜘蛛精", 23, 169.7));
        students.add(new Student("紫霞", 22, 169.8));
        students.add(new Student("至尊宝", 26, 165.5));
        students.add(new Student("牛魔王", 22, 183.5));
        System.out.println(students);
    }
}

在compare中颠倒下位置,就可以实现降序排列。

到此这篇关于Java LinkedHashSet集合的底层原理和TreeSet集合的文章就介绍到这了,更多相关Java LinkedHashSet和TreeSet集合内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java 中通过 key 获取锁的方法

    Java 中通过 key 获取锁的方法

    这篇文章主要介绍了Java 中通过 key 获取锁,本文演示如何对某个 key 加锁,以保证对该 key 的并发操作限制,可以实现同一个 key 一个或者多个线程同时执行,需要的朋友可以参考下
    2022-11-11
  • 详解SpringBoot如何创建自定义Starter

    详解SpringBoot如何创建自定义Starter

    Spring Boot的自动配置机制为开发人员提供了一种轻松集成和配置各种功能的便捷方式,本文将深入探讨在Spring Boot中如何创建自定义Starter,为构建模块化且易维护的应用提供有力的支持,需要的朋友可以参考下
    2024-02-02
  • Java开发实现人机猜拳游戏

    Java开发实现人机猜拳游戏

    这篇文章主要为大家详细介绍了Java开发实现人机猜拳游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-08-08
  • SpringBoot使用validation做参数校验说明

    SpringBoot使用validation做参数校验说明

    这篇文章主要介绍了SpringBoot使用validation做参数校验说明,首先通过添加hibernate-validator展开全文内容,具有一定的参考价值,需要的小伙伴可以参考与喜爱
    2022-04-04
  • Java陷阱之慎用入参做返回值详解

    Java陷阱之慎用入参做返回值详解

    这篇文章主要给大家介绍了关于Java陷阱之慎用入参做返回值的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-12-12
  • springboot配置多数据源的实例(MongoDB主从)

    springboot配置多数据源的实例(MongoDB主从)

    下面小编就为大家分享一篇springboot配置多数据源的实例(MongoDB主从),具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2017-12-12
  • Java 后端接收Request请求参数的多种方式汇总

    Java 后端接收Request请求参数的多种方式汇总

    本文给大家总结了后端接收Request请求参数的7种方式,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
    2023-10-10
  • Java8新特性之默认方法(default)浅析

    Java8新特性之默认方法(default)浅析

    这篇文章主要介绍了Java8新特性之默认方法(default)浅析,默认方法也称为虚拟扩展方法或防护方法,可以让我们修改接口而不破坏原来的实现类的结构,需要的朋友可以参考下
    2014-06-06
  • Mybatisplus详解如何注入自定义的SQL

    Mybatisplus详解如何注入自定义的SQL

    mybatis-plus 提供了许多默认单表 CRUD 语句,对于其他 SQL 情况爱莫能助。如果有一个删库跑路,并且需要多次调用,来清空多张表数据得需求,那么如何把他封装在 mybatis-plus 中调用呢,下面我们一起来看一下
    2022-06-06
  • 堆排序实例(Java数组实现)

    堆排序实例(Java数组实现)

    下面小编就为大家分享一篇使用Java数组实现堆排序的实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2017-12-12

最新评论