【问题标题】:HashSet not removing existing elementHashSet 不删除现有元素
【发布时间】:2026-01-23 09:50:01
【问题描述】:

我有一个输出类,它基本上包含一个 BitSet,它覆盖了 hashCode 和 equals。然后我有一个输出的 HashSet 并执行以下操作:

Set<Output> outputs = new HashSet<>();
Output o1 = new Output();
o1.flip(3);
Output o2 = new Output();
o2.flip(1);
o2.flip(3);
outputs.add(o1);
outputs.add(o2);

如果我进行打印(输出),我会得到 ​​p>

[Output@5a1, Output@5a3]

如果我这样做了

o2.flip(1);

我明白了

[Output@5a3, Output@5a3]

这当然是 Set 的正常行为,因为 Set 无法知道元素的哈希码已更改。

如果我现在这样做

outputs.remove(o1);

我明白了

[Output@5a3]

完美!

如果我再做一次

outputs.remove(o1); //or outputs.remove(o2);

它返回false,而我还有[Output@5a3]

这很奇怪,因为如果我这样做了

outputs.contains(o1) -> false

这可以解释删除行为,虽然我不明白为什么它返回 false,因为如果我这样做了

    for(Output o : outputs) {
        System.out.println(o.equals(o1));
    }

它输出true

任何想法为什么会发生这种情况?

完整代码:

class Output {
    BitSet values;

    public Output() {
        values = new BitSet(4);
    }

    public void flip(int index) {
        values.flip(index);
    }

    public int hashCode() {
        int hash = 3;
        hash = 67 * hash + Objects.hashCode(this.values);
        return hash;
    }

    @Override
    public boolean equals(Object obj) {
        if (!(obj instanceof Output)) {
            return false;
        }
        Output other = (Output) obj;
        return this.values.equals(other.values);
    }
}
public class Main {
    public static void main(String args[]) {
        Set<Output> outputs = new HashSet<>();
        Output o1 = new Output();
        o1.flip(3);
        Output o2 = new Output();
        o2.flip(1);
        o2.flip(3);
        outputs.add(o1);
        outputs.add(o2);
        System.out.println(outputs);
        o2.flip(1);
        System.out.println(outputs);
        outputs.remove(o1);
        System.out.println(outputs);
        outputs.remove(o1);
        System.out.println(outputs);
        for (Output o : outputs) {
            System.out.println(o.equals(o1));
        }
    }
}

输出:

[Output@5a1, Output@5a3]
[Output@5a3, Output@5a3]
[Output@5a3]
[Output@5a3]
true

【问题讨论】:

    标签: java hash set equals


    【解决方案1】:

    当您更改 HashSet 的元素(或 HashMap 中的键)时,该元素的 hashCode 可能会更改(在您的示例中,hashCode 取决于 hashCode BitSet 成员,您已更改)。

    但是,HashSet 不知道该更改,因此不会将元素移动到与新 hashCode 对应的 bin。

    因此,当您搜索该元素时,将使用新的 hashCode 执行搜索(HashSet 搜索始终以 hashCode 开头 - 仅在找到包含具有该 hashCode 的所有元素的 bin 之后, equals() 用于找到正确的元素),但它失败了,因为该元素仍然位于与原始 hashCode 匹配的 bin 中。

    这就是改变 HashSet 的元素是个坏主意的原因。

    【讨论】:

    • “这就是改变 HashSet 的元素是个坏主意的原因。” 至少你应该排除可能从 hascode 计算中改变的属性...