java - “直接 self 参照导致循环” jackson 和继承

标签 java jpa inheritance jackson jaxb

我有一个类似这样的类(class):

@Entity
@Inheritance(strategy = InheritanceType.JOINED)
@Audited
@DiscriminatorColumn(name = "tipo_hito")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(propOrder = { "idHito" })
@XmlRootElement(name = "Hito")
public abstract class Hito implements Ordenable {
    @Id
    @GeneratedValue(strategy= GenerationType.IDENTITY)
    @Column(name = "idhito", nullable = false)
    private Long idHito;

    public Long getIdHito() { return idHito; }

    public void setIdHito(Long idHito) { this.idHito = idHito; }
}

还有一个扩展上面类的类

@MappedSuperclass
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(propOrder = {..., "successors", "predecessors", ...})
@XmlRootElement(name = "HitoOrdenable")
public abstract class HitoOrdenable extends Hito {

    @XmlElement(name = "predecessors")
    @ManyToMany(fetch = FetchType.LAZY)
    @JoinTable(name = "succession", joinColumns = { @JoinColumn(name = "idpredecessor") }, inverseJoinColumns = { @JoinColumn(name = "idsuccessor") })
    private Set<Hito> predecessors;

    @XmlElement(name = "successors")
    @ManyToMany(fetch = FetchType.LAZY)
    @JoinTable(name = "succession", joinColumns = { @JoinColumn(name = "idsuccessor") }, inverseJoinColumns = { @JoinColumn(name = "idpredecessor") })
    private Set<Hito> successors;

    // More code...
}

现在,当我尝试获取 HitoOrdenable 时出现错误。因为例如我可以有:

Hito1
  Hito2-SuccessorOf1
  Hito3-SuccessorOf1
    Hito4-SuccessorOf3
Hito5
  Hito6-SuccessorOf5

所以,当我得到 Hito1 时,它有后继 Hito2Hito3。但是当我得到 Hito2 时,它已经有了前身 Hito1。于是开始“无限循环”。

确切的错误是:

com.fasterxml.jackson.databind.JsonMappingException: Direct self-reference leading to cycle (through reference chain: ar.gob.buenosaires.esb.domain.message.ProyectoRespMsg["proyectos"]->java.util.ArrayList[5]->ar.gob.buenosaires.domain.Proyecto["obras"]->org.hibernate.collection.internal.PersistentBag[0]->ar.gob.buenosaires.domain.Obra["hitos"]->org.hibernate.collection.internal.PersistentBag[0]->ar.gob.buenosaires.domain.HitoObra["hitoOrdenable"])

我尝试在前辈中设置@JsonIgnore,但它不起作用。

最佳答案

经过几天的尝试。我们找到的唯一解决方案是添加 @JsonIgnoreProperties({"predecessors"})@JsonIdentityInfo(generator=ObjectIdGenerators.IntSequenceGenerator.class, property="idHitoAux")

HitoOrdenable.java

  @MappedSuperclass
  @XmlType(propOrder = {..., "successors", "predecessors", ...})
  @XmlAccessorType(XmlAccessType.FIELD)
  @JsonIgnoreProperties({ "predecessors" })
  @JsonIdentityInfo(generator=ObjectIdGenerators.IntSequenceGenerator.class, property="idHitoAux")

Hito.java

@Entity
@Inheritance(strategy = InheritanceType.JOINED)
@Audited
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(propOrder = { "idHito", "tipoHito"})
@JsonIgnoreProperties({"idHitoAux", "hibernateLazyInitializer", "handler"})
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.EXISTING_PROPERTY, visible = true, property = "tipoHito")
@JsonSubTypes({
        @JsonSubTypes.Type(name = "hito_obra", value = HitoObra.class),
        @JsonSubTypes.Type(name = "hito_proyecto", value = HitoProyecto.class)
})<

使用 JsonIgnoreProperties,我们可以防止后继者和前任者之间的无限循环,使用 @JsonIdentityInfo 是因为如果我们使用 "idHito" 而不是 "idHitoAux" code> (它等于 idHito)Hibernate 将 idHito 转换为 null。因此,通过 "idHitoAux" Hibernate 可以识别列表中的每个对象。

PD:在该解决方案之后,我们遇到了抽象类 Hito 的反序列化问题,因此我们的解决方案是添加 @JsonSubTypes

您好!

关于java - “直接 self 参照导致循环” jackson 和继承,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51088693/

相关文章:

java - MultipartFile 在 RestTemplate 中失败,错误请求 400 无正文

java - AsyncSocket延迟发送数据

java - 在预更新事件时获取 hibernate 持久对象

delphi - 将setter添加到后代类的属性中

java - 无法使用 Eclipse 和 Robolectric 解析 Artifact : Missing,

java - 用于外部存储文件的通用图像加载器

java - 使用 LocalDate 从数据库实体检索

Spring, JPA -- 具有许多其他实体传递依赖的实体的 CRUD 集成测试

javascript - 为什么我在原型(prototype)中定义时无法调用函数?

Java super() 继承