HashSet 集合,从入门到精通

开发 前端
在 JDK1.8 中,HashMap 是由数组+链表+红黑树构成,相对于早期版本的 JDK HashMap 实现,新增了红黑树作为底层数据结构,在数据量较大且哈希碰撞较多时,能够极大的增加检索的效率。

哈喽,大家好,我是指北君。

本篇文章指北君给大家介绍一个集合——HashSet,这也是现在面试必问的一个集合,通常用于存放不能重复的数据,可以说这个集合是HashMap 的儿子,为什么这样说呢?看完你就明白了

在 JDK1.8 中,HashMap 是由 数组+链表+红黑树构成,相对于早期版本的 JDK HashMap 实现,新增了红黑树作为底层数据结构,在数据量较大且哈希碰撞较多时,能够极大的增加检索的效率。了解 HashMap 的具体实现后,我们再来介绍由 HashMap 作为底层数据结构实现的一种数据结构——HashSet。(如果不了解 HashMap 的实现原理,建议先看看 HashMap,不然直接看 HashSet 是很难看懂的)。

1.HashSet 定义

HashSet 是一个由 HashMap 实现的集合。元素无序且不能重复。

 public class HashSet<E>
extends AbstractSet<E>
implements Set<E>, Cloneable, java.io.Serializable

图片

和前面介绍的大多数集合一样,HashSet 也实现了 Cloneable 接口和 Serializable 接口,分别用来支持克隆以及支持序列化。还实现了 Set 接口,该接口定义了 Set 集合类型的一套规范。

2.字段属性

 //HashSet集合中的内容是通过 HashMap 数据结构来存储的
private transient HashMap<E,Object> map;
//向HashSet中添加数据,数据在上面的 map 结构是作为 key 存在的,而value统一都是 PRESENT
private static final Object PRESENT = new Object();

第一个定义一个 HashMap,作为实现 HashSet 的数据结构;第二个 PRESENT 对象,因为前面讲过 HashMap 是作为键值对 key-value 进行存储的,而 HashSet 不是键值对,那么选择 HashMap 作为实现,其原理就是存储在 HashSet 中的数据 作为 Map 的 key,而 Map 的value 统一为 PRESENT(下面介绍具体实现时会了解)。

3.构造函数

①无参构造

     public HashSet() {
map = new HashMap<>();
}

直接 new 一个 HashMap 对象出来,采用无参的 HashMap 构造函数,具有默认初始容量(16)和加载因子(0.75)。

②指定初始容量

     public HashSet(int initialCapacity) {
map = new HashMap<>(initialCapacity);
}

③指定初始容量和加载因子

public HashSet(int initialCapacity, float loadFactor) {
map = new HashMap<>(initialCapacity, loadFactor);
}

④构造包含指定集合中的元素


public HashSet(Collection<? extends E> c) {
map = new HashMap<>(Math.max((int) (c.size()/.75f) + 1, 16));
addAll(c);
}

集合容量很好理解,这里我介绍一下什么是加载因子。在 HashMap 中,能够存储元素的数量就是:总的容量*加载因子 ,新增一个元素时,如果HashMap集合中的元素大于前面公式计算的结果了,那么就必须要进行扩容操作,从时间和空间考虑,加载因子一般都选默认的0.75。

4.添加元素

     public boolean add(E e) {
return map.put(e, PRESENT)==null;
}

通过 map.put() 方法来添加元素,在上一篇博客介绍该方法时,说明了该方法如果新插入的key不存在,则返回null,如果新插入的key存在,则返回原key对应的value值(注意新插入的value会覆盖原value值)。

也就是说 HashSet 的 add(E e) 方法,会将 e 作为 key,PRESENT 作为 value 插入到 map 集合中,如果 e 不存在,则插入成功返回 true;如果存在,则返回false。

5.删除元素

     public boolean remove(Object o) {
return map.remove(o)==PRESENT;
}

调用 HashMap 的remove(Object o) 方法,该方法会首先查找 map 集合中是否存在 o ,如果存在则删除,并返回该值,如果不存在则返回 null。

也就是说 HashSet 的 remove(Object o) 方法,删除成功返回 true,删除的元素不存在会返回 false。

6.查找元素

    public boolean contains(Object o) {
return map.containsKey(o);
}

调用 HashMap 的 containsKey(Object o) 方法,找到了返回 true,找不到返回 false。

7.遍历元素

HashSet<Integer> set = new HashSet<>();
set.add(1);
set.add(2);
//增强for循环
for(Integer i : set){
System.out.println(i);
}
//普通for循环
Iterator<Integer> iterator = set.iterator();
while (iterator.hasNext()){
System.out.println(iterator.next());
}

8.小结

好了,这就是JDK中java.util.HashSet 类的介绍。

我是指北君,操千曲而后晓声,观千剑而后识器。感谢各位人才的:点赞、收藏和评论,我们下期更精彩!

责任编辑:武晓燕 来源: Java技术指北
相关推荐

2010-02-06 15:31:18

ibmdwAndroid

2009-07-22 14:55:16

ibmdwAndroid

2016-12-08 22:39:40

Android

2017-05-09 08:48:44

机器学习

2012-02-29 00:49:06

Linux学习

2010-11-08 10:20:18

2024-02-26 08:52:20

Python传递函数参数参数传递类型

2022-09-02 15:11:18

开发工具

2009-07-03 18:49:00

网吧综合布线

2023-10-13 08:23:05

2011-10-26 20:47:36

ssh 安全

2009-03-19 13:36:53

SSH安全通道远程

2024-06-07 08:51:50

OpenPyXLPythonExcel文件

2017-01-09 09:34:03

Docker容器传统虚拟机

2024-01-11 09:35:12

单元测试Python编程软件开发

2023-05-09 08:34:51

PythonWith语句

2010-09-25 11:51:39

无线路由器

2018-06-12 11:01:55

HBase误删数据

2011-06-07 10:33:11

Android Activity
点赞
收藏

51CTO技术栈公众号