Java集合HashSet中的两个对象怎样算重复

本文深入探讨了HashSet的工作原理,包括对象的无序性和不可重复性特点。详细解释了两个对象被视为重复的标准,即内容和hashcode值都必须相同,并讨论了equals()与hashCode()方法的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

集合HashSet中对象的特点是无序、不允许重复。无序好理解,那么两个什么样的对象算重复呢?

两个对象重复意味着这两个对象的内容相同hashcode值也相同

1两个对象AB内容相同,表示A.equals(B)的值为true

(不重写的话,默认equals()方法是调用=="进行判断的,=="判断的是两个对象的引用是否相同)

2如果你不重写hashCode()方法的话,两个对象的hashcode值在默认情况下是不可能相同的。

hashCode()方法继承自Object类。每个对象在创建的时候会被分配一个唯一的hashcode值(一个整形的值)。所以不重写hashCode()方法的话,两个对象的hashcode值是不会相同的。

注意StringhashCode()方法已经被重写,所以两个不同的String对象的hashcode值有可能相同,比如下面的式子的最终结果将是true,虽然str0str是两个不同的对象。

  1. String str0 = new String("Hello!"); 
  2. String str = new String("Hello!"); 
  3. System.out.println(str0.hashCode()==str.hashCode()); 
String str0 = new String("Hello!");
String str = new String("Hello!");
System.out.println(str0.hashCode()==str.hashCode());


只有同时满足以上两个条件的对象,才能算的上重复,这两个对象在加入到HashSet中的时候只能添加进一个。

两个hashcode值相等的对象,equals()返回值不一定为true;而如果两个对象的equals方法返回值为true,则两个对象的hashcode值必须相同,也即如果你重写equals()方法的话,则你必须重写hashCode()方法。

另外:还有一个我们经常用到的是判断两个对象的引用是否相等,即使用”=="符号进行判断,引用表示的是对象在内存(堆)中的存放地址,很显然两个不同的对象的地址是不可能相等的,即使两个对象的值相等(equals()结果返回true)。例如下面的例子讲分别返回true和false。

  1. String str0 = new String("Hello!");   
  2. String str = new String("Hello!");   
  3. System.out.println(str0.equals(str)); 
  4. System.out.println(str0==str); 
String str0 = new String("Hello!");  
String str = new String("Hello!");  
System.out.println(str0.equals(str));
System.out.println(str0==str);

真绕啊!

 

转自:yaozq的android开发

### HashSet 的基本概念 HashSetJava 集合框架中的一个实现类,其底层实际上是通过 HashMap 来实现的,这意味着它的底层数据结构是数组、链表以及红黑树的组合[^2]。这种结构使得 HashSet 在读取和查找操作上具有较高的效率。 ### HashSet 的主要特性 - **唯一性**:HashSet 不允许重复元素的存在,这是通过内部的 HashMap 来保证的,当两个对象的 `equals` 方法返回 true 并且它们的 `hashCode` 相同时,认为这两个对象是相等的。 - **允许 null 元素**:可以向 HashSet 中添加一个 null 元素。 - **非线程安全**:HashSet 不是一个线程安全的集合类,如果需要在多线程环境中使用,应该采取额外的同步措施。 ### HashSet 的性能特点 - **内存开销**:相比于 ArrayList,HashSet 的内存开销更大,因为它基于 HashMap 实现。 - **增删查的时间复杂度**:对于 HashSet 而言,增加、删除和查找元素的时间复杂度均为 O(1),这表明这些操作的平均时间是常数级别的[^3]。 ### 使用场景 由于 HashSet 提供了快速的查找和插入功能,它非常适合用于需要快速访问元素的场景。例如,在需要检查某个元素是否存在的情况下,使用 HashSet 可以大大提高程序的效率。 ### 示例代码 以下是一个简单的 HashSet 使用示例: ```java import java.util.HashSet; public class HashSetExample { public static void main(String[] args) { // 创建一个 HashSet HashSet<String> set = new HashSet<>(); // 添加元素 set.add("Apple"); set.add("Banana"); set.add("Cherry"); // 尝试添加重复元素 boolean isAdded = set.add("Apple"); // 返回 false,因为 "Apple" 已经存在 // 输出结果 System.out.println("Set contains Apple: " + isAdded); System.out.println("Set size: " + set.size()); } } ``` ### 注意事项 - 在使用 HashSet 时需要注意其非线程安全的性质,如果在多线程环境中使用,应该考虑使用同步机制。 - HashSet 中的元素是无序的,这意味着迭代顺序可能与插入顺序不同。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值