有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

java cyclic equals()导致堆栈溢出

我正在使用hibernate并具有双向关系。如何正确重写这两个类的equals()

下面是代码(使用guava对象):(PS:这是一个糟糕的示例,是对实体的选择,但我有兴趣学习推荐的方法)

目的地

@Entity
@Table(name = "DESTINATION")
public class Destination{
    private Integer id;
    private String name;
    private Set<DestinationAlias> aliases = new HashSet<DestinationAlias>(0);


    @Override
    public boolean equals(Object obj) {
        if(obj == this) return true;

        if(obj instanceof Destination){
            final Destination otherDestination = (Destination) obj;
            return Objects.equal(getName().toUpperCase(), otherDestination.getName().toUpperCase()) 
                    && Objects.equal(getAliases(), otherDestination.getAliases());
        }
        return false;
    }
}

作为

@Entity
@Table(name = "DESTINATIONALIAS")
public final class DestinationAlias {
    private Integer idDestinationAlias;
    private String alias;   
    private Destination mainCity;

    @Override
    public boolean equals(Object obj) {
        if(obj == this) return true;

        if(obj instanceof DestinationAlias){
            final DestinationAlias otherAlias = (DestinationAlias) obj;
            return Objects.equal(getAlias().toUpperCase(), otherAlias.getAlias().toUpperCase())
                    && Objects.equal(getMainCity(), otherAlias.getMainCity());
        }
        return false;
    }
}

这是测试用例:

@Test
public void testEqualsto(){
    Destination dest = new Destination("abc", 1.0f, 1.0f);
    dest.getAliases().add(new DestinationAlias("abc alias", dest));

    Destination dest1 = new Destination("abc", 1.0f, 1.0f);
    dest1.getAliases().add(new DestinationAlias("abc alias", dest1));

    assertEquals(dest, dest1);
}

正如所料,stackoverflow会发生,因为每个equals()依次调用另一个equals(),并发生一个循环

为双向实体重写equals()的推荐方法是什么


共 (1) 个答案

  1. # 1 楼答案

    我们必须手动解除冲突。在DestinationAlias中,我将更改equals表达式,以便它仅比较目标ID(它们应该是唯一的):

    return Objects.equal(getAlias().toUpperCase(), otherAlias.getAlias().toUpperCase())
                && Objects.equal(getMainCity().getId(), otherAlias.getMainCity().getId());
                                              ^^^^^^^^                          ^^^^^^^^
    

    进一步阅读