JavaSE学习进阶day06_02 Set集合和Set接口

news2025/1/16 18:54:41

第二章 Set系列集合和Set接口

Set集合概述:前面学习了Collection集合下的List集合,现在继续学习它的另一个分支,Set集合。

set系列集合的特点:

Set接口:

java.util.Set接口和java.util.List接口一样,同样继承自Collection接口,它与Collection接口中的方法基本一致,并没有对Collection接口进行功能上的扩充,只是比Collection接口更加严格了。与List接口不同的是,Set接口都会以某种规则保证存入的元素不出现重复

Set集合有多个子类,这里我们介绍其中的java.util.HashSetjava.util.LinkedHashSetjava.util.TreeSet这两个集合。

tips:Set集合取出元素的方式可以采用:迭代器、增强for。

2.1 HashSet集合介绍

java.util.HashSetSet接口的一个实现类,它所存储的元素是不可重复的,并且元素都是无序的(即存取顺序不能保证不一致)java.util.HashSet底层的实现其实是一个java.util.HashMap支持,由于我们暂时还未学习,先做了解。

HashSet是根据对象的哈希值来确定元素在集合中的存储位置,因此具有良好的存储和查找性能。保证元素唯一性的方式依赖于:hashCodeequals方法

我们先来使用一下Set集合存储,看下现象,再进行原理的讲解:

public class HashSetDemo {
    public static void main(String[] args) {
        //创建 Set集合
        HashSet<String>  set = new HashSet<String>();
​
        //添加元素
        set.add(new String("cba"));
        set.add("abc");
        set.add("bac"); 
        set.add("cba");  
        //遍历
        for (String name : set) {
            System.out.println(name);
        }
    }
}

输出结果如下,说明集合中不能存储重复元素:

cba
abc
bac

tips:根据结果我们发现字符串"cba"只存储了一个,也就是说重复的元素set集合不存储。 

为了方便理解,我们再给出一个例子:

我们发现重复的的第二个“111”字符串没有被添加进去,说明HashSet的特点就是像上面所说的,有去重复的功能,或者说不能添加重复的数据。因此打印只打印了一个字符串:“111”

再看:

添加的顺序是"111"、"333"、"222",但是打印的顺序是"111"、"222"、"333",说明打印顺序和添加顺序没有关系,也就是存的顺序和取的顺序不一致。也就是说HashSet是无序的。其实在底层是用HashMap实现的,后面会细说。

至于无索引怎么体现呢,可以到JDK参考文档里查看,HashSet集合里没有获取索引的方法。因此这也是它的一个特点:无索引。

HashSet在开发中的使用场景:

可以利用HashSet的特性给数据去重

代码示例:

//练习需求:给ArrayList进行去重
ArrayList<String> list = new ArrayList<>();
list.add("aaa");
list.add("aaa");
list.add("bbb");
list.add("bbb");
list.add("ccc");
list.add("ccc");
System.out.println(list);
​
​
//可以利用HashSet的特性给数据去重
HashSet<String> hs = new HashSet<>();
//把list集合中所有的元素全部添加到hs当中
hs.addAll(list);
​
//无序
System.out.println(hs);

运行结果:

2.2 HashSet集合存储数据的结构(哈希表)

什么是哈希表呢?在了解哈希表之前先得知道哈希值:

JDK1.7的哈希值:

关于更多哈希值的知识点请查看课件PPT的内容。

JDK1.8之前,哈希表底层采用数组+链表实现,即使用数组处理冲突,同一hash值的链表都存储在一个数组里。但是当位于一个桶中的元素较多,即hash值相等的元素较多时,通过key值依次查找的效率较低。而JDK1.8中,哈希表存储采用数组+链表+红黑树实现,当链表长度超过阈值(8)时,且数组长度大于等于64时,将链表转换为红黑树,这样大大减少了查找时间

简单的来说,哈希表是由数组+链表+红黑树(JDK1.8增加了红黑树部分)实现的,如下图所示。

看到这张图就有人要问了,这个是怎么存储的呢?

为了方便大家的理解我们结合一个存储流程图来说明一下:

总而言之,JDK1.8引入红黑树大程度优化了HashMap的性能,那么对于我们来讲保证HashSet集合元素的唯一,其实就是根据对象的hashCode和equals方法来决定的如果我们往集合中存放自定义的对象,那么保证其唯一,就必须重写hashCode和equals方法建立属于当前对象的比较方式

哈希表详细流程:

2.3 HashSet存储自定义类型元素

给HashSet中存放自定义类型元素时,比如学生类型,需要重写对象中的hashCode和equals方法,建立自己的比较方式,才能保证HashSet集合中的对象唯一.

创建自定义Student类:

public class Student {
    private String name;
    private int age;
​
​
    public Student() {
    }
​
    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
​
    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }
​
    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }
​
    /**
     * 获取
     * @return age
     */
    public int getAge() {
        return age;
    }
​
​
​
    /**
     * 设置
     * @param age
     */
    public void setAge(int age) {
        this.age = age;
    }
​
​
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Student student = (Student) o;
        return age == student.age && Objects.equals(name, student.name);
    }
​
    @Override
    public int hashCode() {
        return Objects.hash(name, age);
    }
​
    public String toString() {
        return "Student{name = " + name + ", age = " + age + "}";
    }
}
​

创建测试类:

public class HashSetDemo1 {
    public static void main(String[] args) {
        //创建集合
        HashSet<Student> hs = new HashSet<>();
​
        //如果在Student类中,没有重写hashCode和equals方法
        //都是操作的都是地址值。
        //如果我们认为,属性相同,就是同一个对象,那么就需要重写hashCode和equals方法
        //添加元素
        hs.add(new Student("zhangsan",23));
        hs.add(new Student("zhangsan",23));
        hs.add(new Student("zhangsan",23));
​
        System.out.println(hs);
    }
}

这段代码可能理解不透彻,我细说一下,现在就是创建一个学生类Student类,学生属性有姓名跟年龄,我们用快捷键生成标准javabean类后,如下:

注意啊,现在没有重写equals方法和hashCode方法,我们在测试类写入如下代码并运行:

发现重复的元素也添加进去了,这是为什么??不是说HashSet不是有不重复的特点吗?为啥上面那个String就不能添加重复的元素,而自定义的Student类不能。这就是我们要讨论的问题。当添加的元素是自己自定义的类型后,就需要重写equals方法和hashCode方法。快捷键:alt+insert

 

原来这个快捷键每一项都是有用的,现在学了很多了。重写后我们再运行上面那个代码:

 

发现现在不能添加重复的元素了。总结:再用HashSet集合添加自定义元素(除了java提供的都叫自定义元素)时,都要重写equals和hashCode方法才能保证它的不重复性。前面那个String之所以不能添加重复元素,是因为在String类已经重写equals和hashCode方法了。其他包装类也是如此

2.4 LinkedHashSet

我们知道HashSet保证元素唯一,可是元素存放进去是没有顺序的,那么我们要保证有序,怎么办呢

在HashSet下面有一个子类java.util.LinkedHashSet,它是链表和哈希表组合的一个数据存储结构。它能使得元素存放时是有序的。但是在以后的开发中,LinkedHashSet用的不多,因为它的底层多了一个双向链表,效率有点低。以后用hashSet的比较多。

先看一个代码:

发现添加元素的顺序和打印的顺序是一致的。相比hashSet,LinkHashSet就多了一个有序性

演示代码如下:

public class LinkedHashSetDemo {
    public static void main(String[] args) {
        Set<String> set = new LinkedHashSet<String>();
        set.add("bbb");
        set.add("aaa");
        set.add("abc");
        set.add("bbc");
        Iterator<String> it = set.iterator();
        while (it.hasNext()) {
            System.out.println(it.next());
        }
    }
}
结果:
  bbb
  aaa
  abc
  bbc

2.5 TreeSet集合

1. 特点

TreeSet集合是Set接口的一个实现类,底层依赖于TreeMap,是一种基于红黑树的实现,其特点为:

  1. 元素唯一(不能添加重复的元素)

  2. 元素没有索引

  3. 使用元素的自然顺序对元素进行排序,或者根据创建 TreeSet 时提供的 Comparator比较器 进行排序,具体取决于使用的构造方法:

先看一个案例:

发现Integer打印的是默认升序排序,且不能添加重复的元素。

2.排序方式一:默认方式

默认的排序方式,也叫做自然排序

是让Javabean类实现Comparable接口,重写里面的compareTo方法来实现的。

注意点:

1.Java已经写好的String,Integer已经定义好了默认规则。

String:按照首字母的字典顺序排序,如果首字母一样,则比较第二个字母。

Integer:升序

2.我们自己定义了Javabean类需要指定默认排序规则,否则会报错。

代码示例1(存自定义对象):

package com.itheima.a04treesetdemo;
​
public class Student implements Comparable<Student> {
    private String name;
    private int age;
​
    //...空参...
    //...有参...
    //...get和set方法...
    //...toString方法...
​
    @Override
    public int compareTo(Student o) {
        //按照年龄进行排序
        //this表示当前要添加的元素
        //o:已经在树里面存在的元素
        //如果结果是负数,那么就存左边(降序)
        //如果结果是正数,那么就存右边(升序)
        //如果结果是0,认为现在要添加的元素跟当前元素一直,就不存
​
        //System.out.println("this:" + this);
        //System.out.println("o:" + o);
        return this.age - o.age;
    }
}

代码示例2(存Integer):

案例演示自然排序(20,18,23,22,17,24,19):

public static void main(String[] args) {
    //无参构造,默认使用元素的自然顺序进行排序
    TreeSet<Integer> set = new TreeSet<Integer>();
    set.add(20);
    set.add(18);
    set.add(23);
    set.add(22);
    set.add(17);
    set.add(24);
    set.add(19);
    System.out.println(set);
}
​
控制台的输出结果为:
[17, 18, 19, 20, 22, 23, 24]

3.排序方式二:比较器排序

创建集合对象时,传递Comparator实现类的对象,并重写compare方法。

并在使用的时候,默认用第一种,当第一种不能满足要求的时候,可以用第二种排序方式。

比如,Integer默认升序,如果我想降序排列,就需要用第二种了。

比如,String默认按照字典的顺序排列,如果我想按照字符串的长度排列,就需要用第二种了

案例:

演示比较器排序(20,18,23,22,17,24,19):

public static void main(String[] args) {
    //有参构造,传入比较器,使用比较器对元素进行排序
    TreeSet<Integer> set = new TreeSet<Integer>(new Comparator<Integer>() {
        @Override
        public int compare(Integer o1, Integer o2) {
            //元素前 - 元素后 : 升序
            //元素后 - 元素前 : 降序(C语言库的快速排序算法就是这个原理)
            return o2 - o1;
        }
    });
    set.add(20);
    set.add(18);
    set.add(23);
    set.add(22);
    set.add(17);
    set.add(24);
    set.add(19);
    System.out.println(set);
}
​
控制台的输出结果为:
[24, 23, 22, 20, 19, 18, 17]

2.6练习-存储学生信息按照总分排序

需求:

需求:键盘录入3个学生信息(姓名,语文成绩,数学成绩,英语成绩),按照总分从高到低输出到控制台

隐藏的排序规则:总分一致还得按照语文,数学,英语,姓名这样的顺序排序。

分析

①定义学生类

②创建TreeSet集合对象,通过进行排序

③创建学生对象

④把学生对象添加到集合

⑤遍历集合

注意:

在指定顺序的时候默认按照自然排序方式,当自然排序不能满足我们的要求时,就用比较器排序。

代码示例:

public class Student implements Comparable<Student>{
    private String name;
    private double chinese;
    private double math;
    private double english;
    
    ...空参构造...
    ...带参构造...
    ...get和set方法...
    ...toString方法...    
​
    @Override
    public int compareTo(Student o) {
        //this o
        double sum1 = this.chinese + this.math + this.english;
        double sum2 = o.chinese + o.math + o.english;
        double sum = sum1 - sum2;
​
        //总分一样,看语文
        double result = sum == 0 ? this.chinese - o.chinese : sum;
        //语文一样,看数学
        result = result == 0 ? this.math - o.math : result;
        //学生一样,看英语
        result = result == 0 ? this.english - o.english : result;
        //英语一样,看姓名
        result = result == 0 ? this.name.compareTo(o.name) : result;
        return (int)result;
    }
}
​
​
​
public class TreeSetTest1 {
    public static void main(String[] args) {
        //1.创建集合
        TreeSet<Student> ts = new TreeSet<>();
​
        //2.
        //只要使用TreeSet就一定要指定排序规则。
        //首先默认按照自然排序
        Scanner sc = new Scanner(System.in);
        for (int i = 0; i < 3; i++) {
            System.out.println("请输入学生的姓名");
            String name = sc.next();
            System.out.println("请输入语文成绩");
            double chinese = sc.nextDouble();
            System.out.println("请输入数学成绩");
            double math = sc.nextDouble();
            System.out.println("请输入英语成绩");
            double english = sc.nextDouble();
​
            Student s = new Student(name,chinese,math,english);
            ts.add(s);
        }
        //打印
        System.out.println(ts);
    }
}

这个代码含金量很高,一定要掌握,自己手动写出来!!TreeSet的内容差不多就这些了,掌握到这里就已经不错了!

单列集合总结:

以后单列集合用的最多的是,ArrayList、HashSet、TreeSet。前面两种用的最为频繁。默认用ArrayList,要去重的话用HashSet,如果需要排序用TreeSet(这是今后开发用的实战用到的哦)。

LinkedHashSet和LinkedList基本用不到。

可变参数:什么是可变参数?为什么要引入可变参数?如下:

 

如果我们按照之前的做法就是重写多个方法,这样会非常麻烦,因为你也不知道数据有几个,或者有的人直接创建数组,然后传入数组长度,这样也可以,但是还是有点小丢丢麻烦。可变参数就是可以传入任意个数的参数。

如:

测试类:可以传任意个数的数据,这就是可变参数,它的本质就是用数组实现的。

 

注意:当形参有多个时,可变参数只能写在最后面的位置!!放在前面的话后面的形参会传递不到值

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/443599.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

D. Marcin and Training Camp(思维 + 判断一个数二进制位是否是另一个数的子集)

Problem - D - Codeforces 马辛是他大学里的一名教练。有N个学生想参加训练营。马辛是个聪明的教练&#xff0c;所以他只想派那些能冷静合作的学生参加。 让我们关注一下这些学生。每个学生可以用两个整数ai和bi来描述&#xff1b;bi等于第i个学生的技能水平&#xff08;越高越…

【刷题之路】LeetCode 程序员面试金典 08.03. 魔术索引

【刷题之路】LeetCode 程序员面试金典 08.03. 魔术索引 一、题目描述二、解题1、方法1——暴力法1.1、思路分析1.2、代码实现 2、方法2——二分分治2.1、思路分析2.2、代码实现 一、题目描述 原题连接&#xff1a; 面试题 08.03. Magic Index LCCI 题目描述&#xff1a; 魔术索…

计算机网络考试复习——第三章 3.3

3.3 使用广播信道的数据链路层 3.3.1局域网的数据链路层 局域网最主要的特点&#xff1a; 1.网络为一个单位所拥有&#xff1b; 2.地理范围和站点数目均有限。 局域网具有如下主要优点&#xff1a; 1.具有广播功能&#xff0c;从一个站点可很方便地访问全网。 2.便于系统…

分布式问题,你知道几个?

你好&#xff0c;我是田哥 现在的单体服务是很难应付面试了&#xff0c;必须要把分布式相关技术给讲清楚&#xff0c;否则面试难搞。 下面我们来聊聊&#xff0c;分布式环境下会面临哪些问题。 先来看一下主要内容&#xff1a; 分布式系统中常见的难题包括&#xff1a; 一致性问…

PackageMS 启动

1.PackageMS 相关框架类 2.PackageMS 启动过程 2.1 # SystemServer.java /** 291 * The main entry point from zygote. 292 */ 293 public static void main(String[] args) { 294 new SystemServer().run(); 295 } 上面是SystemServer的主函数。 pr…

RK3568开发板 buildroot编译配置

启扬智能RK3568开发板提供两种文件系统&#xff0c;一种是Debian&#xff0c;另外一种是BuildRoot。在使用过程中&#xff0c;有些客户需要添加属于自己的软件包&#xff0c;所以在此分享启扬RK3568关于BuildRoot编译配置的方法以及相关注意事项。 启扬提供的源码是完整sdk&am…

毕业论文用什么流程图软件比较好?

在写作论文的时候使用流程图&#xff0c;会让我们的论文看起来更加有逻辑。并且流程图的图片都可以在PPT中随意插入以及使用。 基础流程图作为最为基本和简单的的流程图方式&#xff0c;一般不区分用户角色和场景&#xff0c;适用于简单场景&#xff0c;梳理单一的流程情况&am…

nssctf web入门(8)

目录 [SWPUCTF 2021 新生赛]easy_sql [SWPUCTF 2021 新生赛]error 这里通过nssctf的题单web安全入门来写&#xff0c;会按照题单详细解释每题。题单在NSSCTF中。 想入门ctfweb的可以看这个系列&#xff0c;之后会一直出这个题单的解析&#xff0c;题目一共有28题&#xff0c;…

mysql(8.0)_主从复制

1.环境介绍 自己的主机--master 同学的主机--slave2.安装mysql https://blog.csdn.net/weixin_45955039/article/details/130144515?spm1001.2014.3001.55013. 准备工作 3.1在云服务器上添加端口号 3.2关闭防火墙 systemctl stop firewalld setenforce 04.master上的配置 …

数字图像处理笔记 第二章 数字图像基础

第二章 数字图像基础 2.1 视觉感知要素 知己知彼&#xff0c;百战不殆。了解人类的视觉感知&#xff0c;有助于我们学习数字图像处理。本小节中眼睛中图像的形成&#xff0c;让我联想到照相机成像&#xff0c;人眼的亮度适应和辨别让我想到手机屏幕的自动调节亮度。由于这部分…

马尔科夫链HMM

在机器学习算法中&#xff0c;马尔可夫链(Markov chain)是个很重要的概念。马尔可夫链&#xff08;Markov chain&#xff09;&#xff0c;又称离散时间马尔可夫链&#xff08;discrete-time Markov chain&#xff09;&#xff0c;因俄国数学家安德烈马尔可夫&#xff08;俄语&a…

如何排查线上环境CPU100%问题

如何排查线上环境CPU100%问题 cpu爆满的原因可能&#xff1a;1、形成了死锁。2、形成了死循环 1.使用arthas 2.使用命令的方式 死锁的检查&#xff1a; 方式一&#xff1a; 可通过jps或者ps -ef | grep java的命令查看到运行程序的PID使用jstack -l pid 命令查看线程的堆栈信…

代码随想录|day49|动态规划part10● 121. 买卖股票的最佳时机 ● 122.买卖股票的最佳时机II

它来了它来了&#xff01;我们最爱的股票问题它来了&#xff01;&#xff01; 121. 买卖股票的最佳时机 链接&#xff1a;代码随想录 视频链接&#xff1a;动态规划之 LeetCode&#xff1a;121.买卖股票的最佳时机1_哔哩哔哩_bilibili 好难懂&#xff0c;尤其是理解状态的变化以…

openGauss 5.0企业版主从部署,实战狂飙

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&am…

【排序】归并排序(递归和非递归)

归并排序 前言图解大致思路代码实现递归非递归 时间复杂度和空间复杂度 前言 这是我讲的最后一个排序了&#xff0c;归并排序难度不大&#xff0c;也是分治的思想。 归并排序时间复杂度是在N*logN里面还是比较优的&#xff0c;毕竟实现起来的是完全二分的&#xff0c;但是差就…

2023第十届大唐杯省赛心得体会总结

第十届“大唐杯”全国大学生新一代信息通信技术大赛结束&#xff0c;分享一下2023第十届大唐杯省赛的相关经验。 年初研究生组就开始报名了&#xff0c;所以这回也是摩拳擦掌&#xff0c;加上大唐杯的认可度很高&#xff0c;今年的情况只会更卷&#xff0c;需要掌握一定的通信…

java数据结构学习第三期

给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置&#xff08;…

4.17-4.18学习总结

MD5 MD5: 1、压缩性 2、容易计算 3、抗修改性 4、弱抗碰撞 5、强抗碰撞 为什么需要MD5&#xff1f; 存储一些敏感信息的时候&#xff0c;如果不进行加密会出现安全问题。 例如&#xff1a;系统登录的密码&#xff0c;如果数据库中的密码采用明文&#xff0c;一旦数据库泄…

扬帆优配|多路资金扎堆博弈,顶级游资章盟主3天爆买21个亿

4月20日&#xff0c;中科曙光&#xff08;603019.SH&#xff09;盘中稳步拉升&#xff0c;最终收涨9.99%&#xff0c;股价一举刷出了历史新高&#xff0c;达到了54.71元/股&#xff0c;全天的成交额也高达97.43亿元。 盘后发表的龙虎榜数据显现&#xff0c;知名游资章盟主的常用…

Qt Quick - PageIndicator

Q 理论使用总结 一、概述二、简单使用例子1. SwipeView 和 PageIndicator2. StackLayout 和 PageIndicator 三、常用属性四、定制化 一、概述 PageIndicator用于指示含有多个页面的容器中&#xff0c;当前处理活动的页。记住&#xff0c;这个只是指示当前的活动页&#xff0c;…