mplementando equals e hashCode para objetos com referências circulares em Ja
Eu tenho duas classes definidas de forma que ambas contenham referências para o outro objeto. Eles se parecem com isso (isso é simplificado; no meu modelo de domínio real, a classe A contém uma lista de B e cada B tem uma referência ao pai A):
public class A {
public B b;
public String bKey;
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + ((b == null) ? 0 : b.hashCode());
result = prime * result + ((bKey == null) ? 0 : bKey.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (!(obj instanceof A))
return false;
A other = (A) obj;
if (b == null) {
if (other.b != null)
return false;
} else if (!b.equals(other.b))
return false;
if (bKey == null) {
if (other.bKey != null)
return false;
} else if (!bKey.equals(other.bKey))
return false;
return true;
}
}
public class B {
public A a;
public String aKey;
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + ((a == null) ? 0 : a.hashCode());
result = prime * result + ((aKey == null) ? 0 : aKey.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (!(obj instanceof B))
return false;
B other = (B) obj;
if (a == null) {
if (other.a != null)
return false;
} else if (!a.equals(other.a))
return false;
if (aKey == null) {
if (other.aKey != null)
return false;
} else if (!aKey.equals(other.aKey))
return false;
return true;
}
}
OhashCode
eequals
foram gerados pelo Eclipse usando os dois campos de A e B. O problema é que chamar oequals
ouhashCode
método @ em qualquer objeto resulta em umStackOverflowError
, já que ambos chamam @ do outro objeequals
ehashCode
método. Por exemplo, o programa a seguir falhará comStackOverflowError
usando os objetos acima:
public static void main(String[] args) {
A a = new A();
B b = new B();
a.b = b;
b.a = a;
A a1 = new A();
B b1 = new B();
a1.b = b1;
b1.a = a1;
System.out.println(a.equals(a1));
}
Se houver algo inerentemente errado em ter um modelo de domínio definido com relacionamentos circulares dessa maneira, entre em contato. Até onde eu sei, esse é um cenário bastante comum, correto?
Qual é a melhor prática para definirhashCode
eequals
nesse caso? Quero manter todos os campos noequals
, para que seja uma comparação profunda e verdadeira de igualdade no objeto, mas não vejo como posso com esse problema. Obrigado