HashSet似乎没有意识到两个对象是相同的.

eff*_*iss 25 java hashset

我正在尝试使用HashSet来存储我创建的类的对象,但显然相同的对象似乎有两个不同的哈希值,这就是为什么contains方法没有意识到该对象已经在HashSet中.这导致我的程序耗尽堆内存.

我不认为我做错了什么,但无论如何我想要第二意见.我做过类似的操作,之前一切正常,这使得这特别烦人.我很感激任何帮助.

这是我的代码

move1 = new Move(t,s);
if(move1.hashCode()==new Move(t,s).hashCode())
    System.out.println("match");
move2 = new Move(s,t);
moves.add(move1); 
moves.add(move2);
if(moves.contains(new Move(t,s)))
    System.out.println("match found");
Run Code Online (Sandbox Code Playgroud)

这是Move类:

public class Move {
    private int move1;
    private int move2;

    Move(int m1, int m2)
    {
        move1 = m1;
        move2 = m2;
    }

    public String toString()
    {
         return String.valueOf(move1)+" "+String.valueOf(move2);
    }
}
Run Code Online (Sandbox Code Playgroud)

这是我得到的输出

Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
    at java.util.HashMap.addEntry(HashMap.java:797)
    at java.util.HashMap.put(HashMap.java:431)
    at java.util.HashSet.add(HashSet.java:194)
    at makeMove.<init>(makeMove.java:33)
Run Code Online (Sandbox Code Playgroud)

Bal*_*usC 40

您需要覆盖类中的Object#hashCode()方法,Move以使其返回实例hashCode()状态的相同值Move.不要忘记覆盖Object#equals().

也可以看看:


提示:如果你使用像Eclipse这样的IDE ,你也可以自动生成它们.右键单击Move该类的某个位置,选择Source> Generate hashCode()和equals().这是它的样子:

@Override
public int hashCode() {
    final int prime = 31;
    int result = 1;
    result = prime * result + move1;
    result = prime * result + move2;
    return result;
}

@Override
public boolean equals(Object obj) {
    if (this == obj)
        return true;
    if (obj == null)
        return false;
    if (getClass() != obj.getClass())
        return false;
    Move other = (Move) obj;
    if (move1 != other.move1)
        return false;
    if (move2 != other.move2)
        return false;
    return true;
}
Run Code Online (Sandbox Code Playgroud)


nos*_*nos 8

HashSet将基于调用hashCode()和equals()来确定相等性.您尚未实现这些,因此您将从Object继承它们.Object的hashCode和equals方法仅基于引用是否相等.

这就是为什么if(move1.hashCode()==new Move(t,s).hashCode())是假的.move1是一个不同于通过调用new Move(t,s).hashCode()创建的实例的实例.

您需要在Move类中实现hashCode和equals.

例如(虽然可能是非最优的,你可能想要一个null安全等于 - 如果可以,你的IDE会生成它们)

public int hashCode() {
    return move1 ^ move2 +;
}

public boolean equals(Object o) {
  if(!other instanceof Move) 
      return false;

  Move other = (Move)o;

  return other.move1 == move1 && other.move2 == move2;
}
Run Code Online (Sandbox Code Playgroud)