Java一分钟之-集合框架进阶:Set接口与HashSet

本文涉及的产品
实时数仓Hologres,5000CU*H 100GB 3个月
智能开放搜索 OpenSearch行业算法版,1GB 20LCU 1个月
实时计算 Flink 版,5000CU*H 3个月
简介: 【5月更文挑战第10天】本文介绍了Java集合框架中的`Set`接口和`HashSet`类。`Set`接口继承自`Collection`,特征是不允许重复元素,顺序不确定。`HashSet`是`Set`的实现,基于哈希表,提供快速添加、删除和查找操作,但无序且非线程安全。文章讨论了`HashSet`的特性、常见问题(如元素比较规则、非唯一性和线程安全性)以及如何避免这些问题,并提供了代码示例展示基本操作和自定义对象的使用。理解这些概念和注意事项能提升代码效率和可维护性。

在Java集合框架中,Set接口是另一种重要的集合类型,它不允许元素重复,并且元素的顺序是不确定的。HashSetSet接口的一个实现,它使用哈希表来存储元素,提供了快速的添加、删除和查找操作。本文将介绍Set接口和HashSet的基本概念、常见问题、易错点及避免策略,并通过代码示例进行说明。
image.png

一、Set接口概览

Set接口继承自Collection接口,其主要特性是不允许重复元素。Set接口没有定义特定的元素顺序,但某些实现类(如TreeSet)会根据元素的自然排序或比较器来决定顺序。

核心方法

  • add(E element): 添加元素,如果集合中已存在该元素,则不会添加。
  • remove(Object o): 删除指定元素,如果存在。
  • contains(Object o): 判断集合是否包含指定元素。
  • isEmpty(): 判断集合是否为空。
  • size(): 获取集合中元素的数量。

二、HashSet介绍

HashSet是基于哈希表实现的Set接口实现,它没有元素顺序,添加元素速度快,但不保证元素的排列顺序。HashSet不允许元素重复,这意味着如果尝试添加已存在的元素,add方法将返回false

特性

  • 快速添加:通过哈希函数快速定位元素,添加效率高。
  • 无序性:元素的顺序是不确定的,不保证添加时的顺序。
  • 非线程安全:与ArrayList类似,HashSet在多线程环境下需额外同步控制。

三、常见问题与易错点

1. 元素比较规则

问题:元素对象未重写equals()hashCode(),导致无法正确判断元素是否重复。 示例

public class User {
   
   
    private String name;

    // ...构造器、getter、setter等省略...
}

Set<User> users = new HashSet<>();
users.add(new User("Alice")); // Alice
users.add(new User("Alice")); // 不会认为是重复

避免:对于自定义对象,确保重写equals()hashCode()方法,以便正确识别相等的实例。

2. 非唯一性

问题:元素的hashCode()方法返回相同值,即使equals()返回false,也可能导致元素被视为重复。 示例

public class User {
   
   
    private int id;

    // ...构造器、getter、setter等省略...
    @Override
    public int hashCode() {
   
   
        return id;
    }
}

Set<User> users = new HashSet<>();
users.add(new User(1)); // User1
users.add(new User(1)); // 不会认为是重复,因为id相同

避免:确保hashCode()方法能根据equals()的结果生成不同的哈希码。

3. 线程安全性

问题:在多线程环境中,多个线程同时修改HashSet可能导致数据不一致。 示例:两个线程同时向HashSet添加元素。 避免:使用线程安全的ConcurrentSkipListSet,或者在多线程环境下对HashSet进行同步控制。

四、代码示例

基本操作

Set<String> names = new HashSet<>();
names.add("Alice"); // true
names.add("Bob");   // true
names.add("Alice"); // false,因为已存在

if (names.contains("Bob")) {
   
   
    names.remove("Bob");
}

for (String name : names) {
   
   
    System.out.println(name);
}

自定义对象的HashSet

public class User {
   
   
    private String name;

    // ...构造器、getter、setter等省略...

    @Override
    public boolean equals(Object obj) {
   
   
        if (this == obj) return true;
        if (obj == null || getClass() != obj.getClass()) return false;
        User user = (User) obj;
        return Objects.equals(name, user.name);
    }

    @Override
    public int hashCode() {
   
   
        return Objects.hash(name);
    }
}

Set<User> users = new HashSet<>();
users.add(new User("Alice"));
users.add(new User("Alice")); // false,因为name相同,被视为重复

五、总结

理解并熟练使用Set接口和HashSet,可以帮助我们更好地组织和管理不重复的数据集。注意元素的比较规则、哈希码的生成,以及在多线程环境下的同步控制,是避免常见问题的关键。合理选择集合类型,结合实际需求,可以提高代码的效率和可维护性。

目录
相关文章
|
9天前
|
前端开发 Java 物联网
智慧班牌源码,采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署
智慧班牌系统是一款基于信息化与物联网技术的校园管理工具,集成电子屏显示、人脸识别及数据交互功能,实现班级信息展示、智能考勤与家校互通。系统采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署与私有化定制。核心功能涵盖信息发布、考勤管理、教务处理及数据分析,助力校园文化建设与教学优化。其综合性和可扩展性有效打破数据孤岛,提升交互体验并降低管理成本,适用于日常教学、考试管理和应急场景,为智慧校园建设提供全面解决方案。
160 70
|
2天前
|
消息中间件 算法 安全
JUC并发—1.Java集合包底层源码剖析
本文主要对JDK中的集合包源码进行了剖析。
|
19天前
|
Java
Java LinkedList集合的深度剖析
总的来说,我希望像说故事一样讲解Java LinkedList集合的使用和实现原理,让有些许枯燥的编程知识变得趣味盎然。在这个“公交车”故事中,你不仅熟悉了LinkedList集合的实现和使用,而且还更深入地理解了数据结构中的链表。链表可能会因为插入和删除的便利性而被选用,虽然它的查找效率并不高,但是在很多场景中仍然十分有效。这就像公交车,虽然它速度不快,但却是城市出行的重要工具。
45 8
|
13天前
|
存储 安全 Java
Java 集合框架详解:系统化分析与高级应用
本文深入解析Java集合框架,涵盖List、Set、Map等核心接口及其常见实现类,如ArrayList、HashSet、HashMap等。通过对比不同集合类型的特性与应用场景,帮助开发者选择最优方案。同时介绍Iterator迭代机制、Collections工具类及Stream API等高级功能,提升代码效率与可维护性。适合初学者与进阶开发者系统学习与实践。
40 0
|
1月前
|
编译器 C++ 容器
【c++丨STL】基于红黑树模拟实现set和map(附源码)
本文基于红黑树的实现,模拟了STL中的`set`和`map`容器。通过封装同一棵红黑树并进行适配修改,实现了两种容器的功能。主要步骤包括:1) 修改红黑树节点结构以支持不同数据类型;2) 使用仿函数适配键值比较逻辑;3) 实现双向迭代器支持遍历操作;4) 封装`insert`、`find`等接口,并为`map`实现`operator[]`。最终,通过测试代码验证了功能的正确性。此实现减少了代码冗余,展示了模板与仿函数的强大灵活性。
64 2
|
2月前
|
编译器 容器
哈希表模拟封装unordered_map和unordered_set
哈希表模拟封装unordered_map和unordered_set
|
2月前
|
编译器 测试技术 计算机视觉
红黑树模拟封装map和set
红黑树模拟封装map和set
|
4月前
|
算法
你对Collection中Set、List、Map理解?
你对Collection中Set、List、Map理解?
109 18
你对Collection中Set、List、Map理解?
|
4月前
|
存储 缓存 安全
只会“有序无序”?面试官嫌弃的List、Set、Map回答!
小米,一位热衷于技术分享的程序员,通过与朋友小林的对话,详细解析了Java面试中常见的List、Set、Map三者之间的区别,不仅涵盖了它们的基本特性,还深入探讨了各自的实现原理及应用场景,帮助面试者更好地准备相关问题。
94 20
|
5月前
|
存储 C++ 容器
【C++】map、set基本用法
本文介绍了C++ STL中的`map`和`set`两种关联容器。`map`用于存储键值对,每个键唯一;而`set`存储唯一元素,不包含值。两者均基于红黑树实现,支持高效的查找、插入和删除操作。文中详细列举了它们的构造方法、迭代器、容量检查、元素修改等常用接口,并简要对比了`map`与`set`的主要差异。此外,还介绍了允许重复元素的`multiset`和`multimap`。
116 3
【C++】map、set基本用法