set集合判断集合中是否有无元素_Java集合框架——Set接口

e4882af81cd42473f120c97c8703d1a2.png

集合框架——Set接口

60b976efe849769efdbd7968d3fc2dbb.png
List集合的特点是有序的,可重复的,是不是存在这一种无序,且能保证元素唯一的集合呢?(HashSet )这就涉及到我们今天所要讲的Set集合
Set可以理解为行为不同的Collection

(一) 概述及功能

(1) 概述

Collection

  • List —— 有序(存储顺序和取出顺序一致),可重复
  • Set—— 无序(存储顺序和取出顺序不一致),唯一
    我们首先要清楚有序无序,到底是什么意思?
    集合所说的序,是指元素存入集合的顺序,当元素存储顺序和取出顺序一致时就是有序,否则就是无序。
    我们一般说的无序是指HashSet,它既不能保证存储和取出顺序一致,更不能保证自然顺序(a-z),而TreeSet 是可以实现自然顺序的。(HashSet的有无序问题可是个大问题,下一篇专篇讲解)

(2) 功能

A:基本功能:(继承而来)

//添加功能

B:特有功能:

//判断元素是否重复,为子类提高重写方法

Set集合中的方法用法并不难,可以参照前面Collection、List集合的讲解,对照学习,我们重点讲解Set中一些重要的特点。

(二) HashSet

一句话记住它:一种没有重复元素的无序集合

我们先说说无序是怎么回事,HashSet 它不保证 set 的迭代顺序,特别是它不保证该顺序恒久不变,也就是说它的存储顺序和取出顺序不一致,虽然说它无序,但是,作为集合来说,它肯定有它自己的存储顺序,而你的顺序恰好和它的存储顺序一致,这代表不了有序。下一篇专篇讲解这一问题!

无序问题由于篇幅较长,我们先放到另一边

我们先来思考一下,HashSet是如何保证不重复的呢?

通过查看HashSet中add方法的源码

// HashSet 源码节选-JKD8

我们可以看到HashSet中add方法所调用的是HashMap中的put方法,我们定位过去

由于解释篇幅较长,直接给出结论,具体源码解释在HashMap源码分析中具体讲解

这个方法底层主要依赖 两个方法:hashCode()和equals()。

  • 步骤:
  • HashSet方法调用add方法时,调用hashCode(),得到一个哈希值,判断哈希值是否相同。
  • 相同:执行equals()方法
    ​ 返回true:说明元素重复,就不添加
    ​ 返回false:说明元素不重复,就添加到集合
  • 不同:就直接把元素添加到集合

现在大家可能想问一句,只使用hashCode()来判断是否重复可以吗?答案是否定的

我们给出这样一句话:

对象相等则hashCode一定相等,hashCode相等对象未必相等,只有equals返回true,hashCode才相等

  • 如果类没有重写这两个方法,默认使用的Object()。一般来说不会相同。
  • 而String类重写了hashCode()和equals()方法,所以,它就可以把内容相同的字符串去掉。只留下一个。
  • 对于String 类型来说,不用重写 hashCode()方法和equals()方法都可以保证元素的唯一性,但是如果不是Stirng,而是其它自定义的对象就要重写这两个方法才能保证元素的唯一性。

(三) TreeSet

概述:

TreeSet:底层是二叉树结构(红黑树是一种自平衡的二叉树)

48124a196324af8ef851b655d3965d1d.png

如何存储

那么这一种结构又是如何存储元素的呢?(我们将上图中圆圈称为节点)

  • 第一个元素存储的时候,直接作为根节点存储
  • 第二个元素开始,每个元素从根节点开始比较
    • 若大 则作为右孩子
    • 若小 则作为左孩子
    • 相等 则不作处理

我们来举一个例子看看:

import 

我们使用图片来解释一下上面的代码

8f6eb08f379f34ebe6f9aac44d16ad16.png

我们将第一个数字20 作为根节点存放,第二个数字18比20小所以放在左边 23大放在右边

例如22这个数字是如何放到如图的位置呢?

首先22先和20比较是大的所以放到右边,接着继续和23进行比较是小的,所以放到23的左边,接下来同理

我们看到运行结果,很神奇的是按照顺序输出的,这也正符合了我们一开始给出的结论:TreeSet 是可以实现自然顺序的

如何取出

那么TreeSet中元素是如何取出来的呢?

从根节点开始,按照左,中,右的原则依次取出元素即可

分析:我们的根节点是20,所以先看左边也就是18,但是下面还有子节点,我们继续看左边所以第一个数字就是17,然后再看中和右也就是18和19,这时候根节点的左边也就全部看完了,所以接着就是中间的根节点20,右边同理。

如何存储自定义对象

我们设定一种场景,存储学生类中的学生对象,并且按照年龄从小到大排序(自然排序)

当满足所有成员变量的值都相同的时候即为同一个元素

注意:如果一个类的元素要想能够进行自然排序,就必须实现自然排序接口(关键)

public 

我们可以专门定义了一个类MyComparator,其实也可以省略这一个类,直接在TreeSetDemo测试类中定义一个匿名内部类

package 

Collection 集合总结

0ec32319b339e153f520dd33ecc4c542.png

Collection集合应用时的选择

是否唯一

  • 唯一:Set
    • 需要排序:TreeSet
    • 不需要排序:HashSet
    • 如果你知道是Set,但是不知道是哪个Set,就用HashSet。
  • 不唯一:List
    • 需要安全:Vector
    • 不需要安全:ArrayList或者LinkedList
    • 查询多:ArrayList
    • 增删多:LinkedList

如果三原则

如果你知道是List,但是不知道是哪个List,就用ArrayList。

如果你知道是Collection集合,但是不知道使用谁,就用ArrayList。

如果你知道用集合,就用ArrayList。

在集合中常见的数据结构

ArrayXxx:底层数据结构是数组,查询快,增删慢

LinkedXxx:底层数据结构是链表,查询慢,增删快

HashXxx:底层数据结构是哈希表。依赖两个方法:hashCode()和equals()

TreeXxx:底层数据结构是二叉树。两种方式排序:自然排序和比较器排序

结尾:

如果内容中有什么不足,或者错误的地方,欢迎大家给我留言提出意见, 蟹蟹大家 !^_^

如果能帮到你的话,那就来关注我吧!(系列文章均会在公众号第一时间更新)

在这里的我们素不相识,却都在为了自己的梦而努力 ❤
一个坚持推送原创Java技术的公众号:理想二旬不止
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值