【问题标题】:Infinite recursion in Hibernate-JPA [duplicate]Hibernate-JPA中的无限递归[重复]
【发布时间】:2020-06-25 13:42:01
【问题描述】:

我在错误之下得到无限递归。

Resolved [org.springframework.http.converter.HttpMessageNotWritableException: Could not write JSON: Infinite recursion (StackOverflowError); nested exception is com.fasterxml.jackson.databind.JsonMappingException: Infinite recursion (StackOverflowError) (through reference chain: org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSDocumentClass$HibernateProxy$u3IXr3xN["groups"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSGroupBean["docClasses"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSDocumentClass$HibernateProxy$u3IXr3xN["groups"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSGroupBean["docClasses"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSDocumentClass$HibernateProxy$u3IXr3xN["groups"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSGroupBean["docClasses"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSDocumentClass$HibernateProxy$u3IXr3xN["groups"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSGroupBean["docClasses"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSDocumentClass$HibernateProxy$u3IXr3xN["groups"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSGroupBean["docClasses"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSDocumentClass$HibernateProxy$u3IXr3xN["groups"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSGroupBean["docClasses"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSDocumentClass$HibernateProxy$u3IXr3xN["groups"]->org.hibernate.collection.internal.PersistentBag[0]->com.dms.bean.DMSGroupBean["docClasses"]]

下面是我的代码

    @Entity
    @Table(name="DMS_Document_Class")
    public class DMSDocumentClass {

        @Id
        @GeneratedValue(strategy=GenerationType.AUTO)
        @Column(name="DocumentClass_Index_ID")
        private long docClassindexID;


        @ManyToMany(fetch = FetchType.LAZY)
        @JoinTable(name = "DOCCLASS_GROUP", 
        joinColumns = {@JoinColumn(referencedColumnName ="DocumentClass_Index_ID")}, inverseJoinColumns = {@JoinColumn(referencedColumnName="groupID")})
        private List<DMSGroupBean> groups;


        @Column(name="Document_Class_Name")
        private String documentClassName;

        @Column(name="Document_Priority")
        private int documentPriority;

        @Column(name="createdDate")
        private Date createdDate;

        @Column(name="createdBy")
        private String createdBy;

另一个类

    @Entity
    @Table(name="DMS_Group", indexes= @Index(columnList="groupName"),
                uniqueConstraints={@UniqueConstraint(columnNames={"groupName"})})
    public class DMSGroupBean {

        @Id
        @GeneratedValue(strategy=GenerationType.AUTO)
        @Column(name="groupID")
        private long groupID;

        @Column(name="groupName")
        private String groupName;

        @Column(name="createDate")
        private Date createDate;

        @ManyToMany(mappedBy = "groups",fetch = FetchType.LAZY)
        private List<DMSUser> users;

        @ManyToMany(mappedBy = "groups",fetch = FetchType.LAZY)
        private List<DMSDocumentClass> docClasses;

        @Transient
        private String status;

我无法理解为什么在 UI 上获取此值时出现此错误。

    @Transactional
        public HashMap<String, Object> getDocListByFolderID(@NonNull long folderID, HttpSession session) {



            DMSFolder currentFolder = folderRepo.findById(folderID).get();

            List<DMSDocument> documents =  documentRepository.findByDocumentFolderID(folderID);

            List<DMSFolder> fetchedSubfolders = folderRepo.findByParentFolderID(folderID);

            //Allow those folder which has rights for this user
            DMSUser userInfo = (DMSUser) session.getAttribute("userinfo");
            List<DMSGroupBean> groups = userInfo.getGroups();//userInfo.getGroupRights();

            List<DMSFolder> finalSubfolders = fetchFinalSubFolders(groups, fetchedSubfolders);

            DMSFolder parentFolder = null;
            if(currentFolder.getParentFolderID() != 0) {
                parentFolder = folderRepo.findById(currentFolder.getParentFolderID()).get();
            }


            HashMap<String, Object> docList = new HashMap<String, Object>();

            docList.put("currentFolder", currentFolder);
            docList.put("documents", documents);
            docList.put("subfolders", finalSubfolders);
            docList.put("parentFolder", parentFolder);

            return docList;
        }

【问题讨论】:

标签: spring hibernate spring-boot jpa spring-data-jpa


【解决方案1】:

您会得到这个StackOverflowException,因为您的 JPA 实体具有双向关系。当 Jackson 尝试反序列化它时,它会跟随您的对象图,该对象图永远不会结束,因为 DMSGroupBeanDMSDocumentClass 正在引用自己。

考虑在关系的一个站点上使用@JsonIgnore,例如:

@Entity
@Table(name="DMS_Group", indexes= @Index(columnList="groupName"),
            uniqueConstraints={@UniqueConstraint(columnNames={"groupName"})})
public class DMSGroupBean {

    @Id
    @GeneratedValue(strategy=GenerationType.AUTO)
    @Column(name="groupID")
    private long groupID;

    @Column(name="groupName")
    private String groupName;

    @Column(name="createDate")
    private Date createDate;

    @ManyToMany(mappedBy = "groups",fetch = FetchType.LAZY)
    private List<DMSUser> users;

    @ManyToMany(mappedBy = "groups",fetch = FetchType.LAZY)
    @JsonIgnore // here
    private List<DMSDocumentClass> docClasses;

    @Transient
    private String status;

}

【讨论】:

  • 这解决了我的问题。谢谢大家。
  • 完美,能否将此答案标记为正确,以便开发者以后搜索相同问题时找到它?
【解决方案2】:

如前所述,导致问题的是双向映射。 JsonIgnore 将解决您的问题,但是,我建议您使用 JsonManagedReference 和 JsonBackReference。不同之处在于 JsonIgnore 会将字段序列化为 null,而 JsonBackReference 将使用其 Id 序列化集合(通过应用于类的 JsonIdentityInfo 注释)。

 @Entity
@Table(name="DMS_Group", indexes= @Index(columnList="groupName"),
            uniqueConstraints={@UniqueConstraint(columnNames={"groupName"})})
@JsonIdentityInfo(generator = ObjectIdGenerators.PropertyGenerator.class,property = "groupID")
public class DMSGroupBean {

   @ManyToMany(mappedBy = "groups",fetch = FetchType.LAZY)
   @JsonManageReference
   private List<DMSDocumentClass> docClasses;

@Entity
@Table(name="DMS_Document_Class")
@JsonIdentityInfo(generator = ObjectIdGenerators.PropertyGenerator.class,property = "docClassindexID")
public class DMSDocumentClass {

    @ManyToMany(fetch = FetchType.LAZY)
    @JoinTable(name = "DOCCLASS_GROUP", 
    joinColumns = {@JoinColumn(referencedColumnName ="DocumentClass_Index_ID")}, inverseJoinColumns = {@JoinColumn(referencedColumnName="groupID")})
    @JsonBackReference
    private List<DMSGroupBean> groups;

【讨论】:

    猜你喜欢
    • 2018-10-02
    • 1970-01-01
    • 1970-01-01
    • 2011-03-20
    相关资源
    最近更新 更多