Set的add方法返回false

本文解决了一个在使用TreeSet存储自定义对象时出现的问题,即当两个对象的比较结果为0时,TreeSet会认为这两个对象是相等的。通过调整compareTo方法的实现,使得即使支持数相同也不会被视为相等,并实现了降序排列。

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

遇到这个问题的时候,第一反应是SQL写错了,因为我是遍历ResultSet调用的Set.add。
后来发现不是SQL的问题,再仔细想了一下,发现了问题所在。
我的Bean实现了Comparable接口,compareTo是这么写的

public int compareTo(GirlInfo o) {
    if(girlSupportNumber > o.getGirlSupportNumber())
        return 1;
    else if(girlSupportNumber == o.getGirlSupportNumber())
        return 0;
    else
        return -1;
}

本身的写法没问题,问题在于我的需求:当两个女生的支持数相同时,不认为他们就是相等的。compareTo不能返回0。代码修改后正常。

public int compareTo(GirlInfo o) {
    if(girlSupportNumber >= o.getGirlSupportNumber())
        return -1;
    else 
        return 1;
}

另外一点,TreeSet默认是升序排列。所以要实现降序的话,在当前对象大的情况下要返回负值。

### Java Set `add()` 方法导致数据丢失的原因及解决方案 `Set` 是 Java 集合框架中用于存储无重复元素的接口,常见的实现类包括 `HashSet`、`TreeSet` 和 `LinkedHashSet`。`Set` 的核心特性是不允许重复元素,因此在调用 `add()` 方法时,如果添加的元素已经存在,新元素将不会被插入。 #### 数据丢失的原因 1. **唯一性约束**:`Set` 的设计目的是保证元素的唯一性。当调用 `add(E e)` 方法时,如果集合中已存在与 `e` 相等(根据 `equals()` 和 `hashCode()` 方法判断)的元素,则新元素不会被添加[^2]。 ```java Set<String> set = new HashSet<>(); set.add("A"); set.add("A"); // 第二次添加不会生效 System.out.println(set.size()); // 输出 1 ``` 2. **自定义对象未正确重写 `equals()` 和 `hashCode()`**:对于自定义对象,如果没有正确重写这两个方法,可能导致 `Set` 无法正确识别重复元素,从而出现“数据丢失”或“重复插入”的现象。 3. **并发修改异常**:在多线程环境下,如果多个线程同时调用 `add()` 方法而未进行同步控制,可能导致元素未被正确插入或抛出 `ConcurrentModificationException`。 #### 解决方案 1. **理解 `Set` 的唯一性机制**:由于 `Set` 的本质是去重,因此在调用 `add()` 时,若元素已存在,应预期到新元素不会被插入。如果需要保留所有元素,应使用 `List` 而非 `Set` [^2]。 2. **正确重写 `equals()` 和 `hashCode()` 方法**:对于自定义类,应确保这两个方法的一致性,以保证 `Set` 能正确识别重复对象。 ```java public class Person { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } @Override public boolean equals(Object o) { if (this == o) return true; if (!(o instanceof Person)) return false; Person person = (Person) o; return age == person.age && name.equals(person.name); } @Override public int hashCode() { return Objects.hash(name, age); } } ``` 3. **使用线程安全的集合实现**:在并发环境中,可以使用 `Collections.synchronizedSet()` 或 `ConcurrentSkipListSet` 等线程安全的 `Set` 实现,以避免并发修改异常。 ```java Set<String> synchronizedSet = Collections.synchronizedSet(new HashSet<>()); ``` 4. **使用 `contains()` 方法检查元素是否存在**:在调用 `add()` 前,可以先使用 `contains()` 方法判断元素是否已存在,以避免不必要的插入操作。 ```java if (!set.contains("A")) { set.add("A"); } ``` 5. **使用 `add()` 方法返回值判断是否插入成功**:`Set.add()` 方法返回一个布尔值,表示是否成功插入元素。可以通过该返回值判断是否发生了“数据丢失”。 ```java boolean added = set.add("A"); if (!added) { System.out.println("元素已存在,未插入"); } ``` ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值