Тройное иерархическое сопоставление сущностей с данными гибернации / пружины вызывает бесконечный рекурсивный цикл - PullRequest
0 голосов
/ 13 сентября 2018

У меня проблемы с проблемой. Итак: у меня есть пять сущностей, у которых есть реляционное сопоставление с другими. Вот как выглядит эта структура:

Диаграмма отношений

Book.java сущность

@JsonManagedReference
@OneToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL,mappedBy = "book")
private Set<Review> reviews = new HashSet<>();
@JsonManagedReference
@OneToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL, mappedBy = "book")
private Set<Comment> comments = new HashSet<>();

@ManyToOne(fetch = FetchType.EAGER, cascade = {CascadeType.MERGE,CascadeType.DETACH,
        CascadeType.PERSIST,CascadeType.REFRESH})
@JoinColumn(name = "author_last_name")
@JsonBackReference
private Author author;

Author.java сущность:

    @JsonManagedReference
@OneToMany(fetch = FetchType.EAGER, cascade = {CascadeType.MERGE,
        CascadeType.DETACH,CascadeType.PERSIST,CascadeType.REFRESH},
        mappedBy = "author")
private Set<Book> books = new HashSet<>();

Review.java сущность

@ManyToOne(fetch = FetchType.LAZY, cascade = {CascadeType.DETACH,CascadeType.MERGE,CascadeType.PERSIST,CascadeType.REFRESH})
@JoinColumn(name = "book_id")
@JsonBackReference
private Book book;

Comment.java сущность

@ManyToOne(fetch = FetchType.LAZY, cascade = {CascadeType.DETACH,CascadeType.MERGE,
        CascadeType.PERSIST,CascadeType.REFRESH})
@JoinColumn(name = "book_id")
@JsonBackReference
private Book book;

В каждой сущности у меня есть getters / setter / hashCode и т. Д. Также есть методы для добавления рецензии на книгу или книги автору.

    public void addBookToAuthor(Book book){
    books.add(book);
    book.setAuthor(this);
}

И проблема в том, что я могу получить всех авторов, одного автора и т. Д., Но когда я пытаюсь установить пару книг для Автора следующим образом:

    public void shouldSaveAnAuthorAndSetBooks(){
        Author author = authorService.findAuthorByLastName("qwerty");
        HashSet<Book> books = new HashSet<>();
        books.add(bookService.findById(5));
        books.add(bookService.findById(8));
        books.add(bookService.findById(10));
        books.forEach(book -> author.addBookToAuthor(book));
        authorRepository.save(author);
}

Это ведет меня к бесконечному циклу, когда в спящем режиме выбираются все книги / обзоры / комментарии:

Hibernate: select reviews0_.book_id as book_id5_4_0_, reviews0_.review_id as review_i1_4_0_, reviews0_.review_id as review_i1_4_1_, reviews0_.book_id as book_id5_4_1_, reviews0_.review as review2_4_1_, reviews0_.review_downvotes as review_d3_4_1_, reviews0_.review_upvotes as review_u4_4_1_ from review reviews0_ where reviews0_.book_id=?

эта строка печатается более 1000 раз. Затем показывает 1000+ предупреждений об отказоустойчивой очистке и, наконец, выбрасывает StackOverFlowExc следующим образом:

java.lang.StackOverflowError
at com.mysql.cj.NativeSession.execSQL(NativeSession.java:1220)
at com.mysql.cj.jdbc.ClientPreparedStatement.executeInternal(ClientPreparedStatement.java:952)
at com.mysql.cj.jdbc.ClientPreparedStatement.executeQuery(ClientPreparedStatement.java:1025)
at com.zaxxer.hikari.pool.ProxyPreparedStatement.executeQuery(ProxyPreparedStatement.java:52)
at com.zaxxer.hikari.pool.HikariProxyPreparedStatement.executeQuery(HikariProxyPreparedStatement.java)
at org.hibernate.engine.jdbc.internal.ResultSetReturnImpl.extract(ResultSetReturnImpl.java:60)
at org.hibernate.loader.plan.exec.internal.AbstractLoadPlanBasedLoader.getResultSet(AbstractLoadPlanBasedLoader.java:419)
at org.hibernate.loader.plan.exec.internal.AbstractLoadPlanBasedLoader.executeQueryStatement(AbstractLoadPlanBasedLoader.java:191)
at org.hibernate.loader.plan.exec.internal.AbstractLoadPlanBasedLoader.executeLoad(AbstractLoadPlanBasedLoader.java:121)
at org.hibernate.loader.plan.exec.internal.AbstractLoadPlanBasedLoader.executeLoad(AbstractLoadPlanBasedLoader.java:86)
at org.hibernate.loader.collection.plan.AbstractLoadPlanBasedCollectionInitializer.initialize(AbstractLoadPlanBasedCollectionInitializer.java:87)
at org.hibernate.persister.collection.AbstractCollectionPersister.initialize(AbstractCollectionPersister.java:688)
at org.hibernate.event.internal.DefaultInitializeCollectionEventListener.onInitializeCollection(DefaultInitializeCollectionEventListener.java:75)
at org.hibernate.internal.SessionImpl.initializeCollection(SessionImpl.java:2223)
at org.hibernate.collection.internal.AbstractPersistentCollection$4.doWork(AbstractPersistentCollection.java:565)
at org.hibernate.collection.internal.AbstractPersistentCollection.withTemporarySessionIfNeeded(AbstractPersistentCollection.java:247)
at org.hibernate.collection.internal.AbstractPersistentCollection.initialize(AbstractPersistentCollection.java:561)
at org.hibernate.collection.internal.AbstractPersistentCollection.read(AbstractPersistentCollection.java:132)
at org.hibernate.collection.internal.PersistentSet.hashCode(PersistentSet.java:430)
at java.util.Arrays.hashCode(Arrays.java:4146)
at java.util.Objects.hash(Objects.java:128)
at com.example.com.example.entity.Book.hashCode(Book.java:190)
at java.util.Arrays.hashCode(Arrays.java:4146)
at java.util.Objects.hash(Objects.java:128)
at com.example.com.example.entity.Review.hashCode(Review.java:93)
at java.util.HashMap.hash(HashMap.java:339)
at java.util.HashMap.put(HashMap.java:612)
at java.util.HashSet.add(HashSet.java:220)
at java.util.AbstractCollection.addAll(AbstractCollection.java:344)
at org.hibernate.collection.internal.PersistentSet.endRead(PersistentSet.java:327)

Конечно, эта трассировка стека тоже показана в цикле. Итак, мой вопрос: что я могу сделать? Должен ли я изменить сопоставления? Или подход к экономии?

1 Ответ

0 голосов
/ 13 сентября 2018

Ну, проблема была тривиальной.Произошла ошибка в переопределенном методе HashCode в объекте book & review.Итак, мораль такова: не используйте HashSet или HashMap и позвольте вашей IDE генерировать для вас хеш-код.Лучше напиши сам

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...