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

本文深入探讨了HashSet的工作原理,包括对象的重复性定义、equals方法与hashCode方法的关系以及它们如何共同确保集合内元素的唯一性。

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

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

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

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

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

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

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

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。

String str0 = new String("Hello!");  
String str = new String("Hello!");  
System.out.println(str0.equals(str));
System.out.println(str0==str);
真绕啊!
Java 中,判断对象集合两个属性值是否重复,可以使用多种方式实现。 ### 方式一:传统的双重循环 通过双重循环遍历集合,比较每两个元素的两个属性值是否都相同。 ```java import java.util.ArrayList; import java.util.List; class Person { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } } public class DuplicateCheck { public static boolean hasDuplicateProperties(List<Person> list) { for (int i = 0; i < list.size(); i++) { for (int j = i + 1; j < list.size(); j++) { Person p1 = list.get(i); Person p2 = list.get(j); if (p1.getName().equals(p2.getName()) && p1.getAge() == p2.getAge()) { return true; } } } return false; } public static void main(String[] args) { List<Person> people = new ArrayList<>(); people.add(new Person("Alice", 25)); people.add(new Person("Bob", 30)); people.add(new Person("Alice", 25)); boolean result = hasDuplicateProperties(people); System.out.println("是否有重复属性: " + result); } } ``` 这种方式简单直接,但时间复杂度为 $O(n^2)$,当集合元素较多时,性能较低。 ### 方式二:使用 Java 8 Stream API 和自定义对象作为键 将两个属性组合成一个自定义对象作为键,使用 `Collectors.groupingBy` 进行分组,然后检查是否有分组的元素数量大于 1。 ```java import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.stream.Collectors; class Person { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } // 自定义对象用于作为键 static class NameAgeKey { private final String name; private final int age; public NameAgeKey(String name, int age) { this.name = name; this.age = age; } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; NameAgeKey that = (NameAgeKey) o; return age == that.age && name.equals(that.name); } @Override public int hashCode() { int result = name.hashCode(); result = 31 * result + age; return result; } } } public class DuplicateCheckStream { public static boolean hasDuplicateProperties(List<Person> list) { Map<Person.NameAgeKey, Long> groups = list.stream() .collect(Collectors.groupingBy(person -> new Person.NameAgeKey(person.getName(), person.getAge()), Collectors.counting())); return groups.values().stream().anyMatch(count -> count > 1); } public static void main(String[] args) { List<Person> people = new ArrayList<>(); people.add(new Person("Alice", 25)); people.add(new Person("Bob", 30)); people.add(new Person("Alice", 25)); boolean result = hasDuplicateProperties(people); System.out.println("是否有重复属性: " + result); } } ``` 这种方式利用了 Stream API 和 `Collectors.groupingBy` 方法,代码更简洁,且性能相对较好,时间复杂度接近 $O(n)$。 ### 方式三:使用 `Set` 存储组合键 将两个属性组合成一个字符串作为键,存储在 `Set` 中,利用 `Set` 的特性判断是否有重复。 ```java import java.util.ArrayList; import java.util.HashSet; import java.util.List; import java.util.Set; class Person { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } } public class DuplicateCheckSet { public static boolean hasDuplicateProperties(List<Person> list) { Set<String> keySet = new HashSet<>(); for (Person person : list) { String key = person.getName() + "-" + person.getAge(); if (!keySet.add(key)) { return true; } } return false; } public static void main(String[] args) { List<Person> people = new ArrayList<>(); people.add(new Person("Alice", 25)); people.add(new Person("Bob", 30)); people.add(new Person("Alice", 25)); boolean result = hasDuplicateProperties(people); System.out.println("是否有重复属性: " + result); } } ``` 这种方式也是 $O(n)$ 的时间复杂度,通过 `Set` 的 `add` 方法的返回值判断是否有重复元素。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值