3 / 3 / 1
Регистрация: 25.12.2014
Сообщений: 63

Spring REST, Hibernate. Сложности десериализации JSON при one-to-many

03.04.2018, 14:47. Показов 3067. Ответов 1

Студворк — интернет-сервис помощи студентам
Всем добрый день.

Возникла проблема, подходящее решение которой никак не нагуглю. Делаю REST API на Spring и Hibernate.
При попытке отправить POST на добавление нового объекта (новой записи в БД) получаю InvalidDefinitionException:
Failed to evaluate Jackson deserialization for type [[simple type, class com.ivt.da.entity.Subject]]: com.fasterxml.jackson.databind.exc.Inval idDefinitionException: Cannot handle managed/back reference 'defaultReference': back reference type (java.util.List) not compatible with managed type (com.ivt.da.entity.Subject)
Предположим, имеется две сущности: школьный предмет и раздел, к которому он относится. Например, предмет "алгебра" относится к разделу "математика и информатика". Очевидно, что не только "алгебра" относится к этому разделу, но и другие предметы. Для этих сущностей построены вот такие POJO-классы:

Класс "Школьные предметы" (Subject)
Кликните здесь для просмотра всего текста

Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
@Entity
@Table(name="subjects")
public class Subject implements Serializable {
 
    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    @Column(name="id")
    private int id;
 
    @ManyToOne(fetch = FetchType.EAGER,
            cascade = {
            CascadeType.DETACH, CascadeType.MERGE,
            CascadeType.PERSIST, CascadeType.REFRESH})
    @JoinColumn(name="section_id")
    @JsonManagedReference
    private SubjectSection section;
 
    @Column(name="name")
    private String name;
 
    public Subject() {
        
    }
 
    // getters and setters
}


Класс "Разделы образовательной программы" (SubjectSections)
Кликните здесь для просмотра всего текста

Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
@Entity
@Table(name="subject_sections")
public class SubjectSection implements Serializable {
 
    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    @Column(name = "id")
    private int id;
 
    @Column(name = "name")
    private String name;
 
    @OneToMany(fetch = FetchType.LAZY, mappedBy = "section", cascade = CascadeType.ALL)
    @JsonBackReference
    private List<Subject> subjects;
 
    public SubjectSection() {
    }
 
    // getters and setters
 
    public void addSubject(Subject subject){
        if(subjects == null)
            subjects = new ArrayList<>();
 
        subjects.add(subject);
        subject.setSection(this);
    }
 
}


Всё это деплоится на Tomcat, куда я отправляю запрос. Запросы всех видов отлично работают для получения-создания-изменения-удаления нового раздела. Вот, например, ответ-список разделов.
Кликните здесь для просмотра всего текста
JSON
1
2
3
4
5
6
7
8
9
10
11
12
13
14
[
    {
        "id": 1,
        "name": "Математика и информатика"
    },
    {
        "id": 2,
        "name": "Руский язык и литература"
    },
    {
        "id": 3,
        "name": "Иностранные языки"
    }
]


Также хорошо работает получение списка предметов и одного предмета по id (в ответе сервера хорошо бы иметь объект раздела целиком, как сейчас и есть на самом деле — нужно для клиента на React). Например по id 1 получаю это:
Кликните здесь для просмотра всего текста
JSON
1
2
3
4
5
6
7
8
{
        "id": 1,
        "section": {
            "id": 1,
            "name": "Математика и информатика"
        },
        "name": "Математика"
    }



Однако, при попытке выполнить POST-запрос на создание нового предмета (Subject) я получаю вышеуказанный эксепшн, связанный с ленивой загрузкой. Тело запроса таково:
JSON
1
2
3
4
5
6
7
{
    "section": {
        "id": 3,
        "name": "Иностранные языки"
    },
    "name": "Английский язык"
}
То есть в section отсутствует список предметов секции (оно и понятно).


Вот контроллер, который это получает:
Кликните здесь для просмотра всего текста
Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
@RestController
@RequestMapping("/subjects")
public class SubjectsApiController {
 
    @Autowired
    private SubjectService subjectService;
 
    // ...
    
    @RequestMapping(value = "/", method = RequestMethod.POST, headers = "Accept=application/json")
    public void addSubject(@RequestBody Subject subject) {
        System.out.println("Controller layer");
        subjectService.saveSubject(subject);
    }
    
    // ...
}


Подскажите, пожалуйста, что можно сделать такого, чтобы:
1) запрос на добавление исполнялся и не падал с ошибкой на этапе десериализации;
2) не испортить ответ по GET-запросам, т.е. предметы должны идти вместе с разделами, а разделы в выдаче не должны иметь предметы.

Всё, что удавалось нагуглить либо не работало, либо ломало пункт 2.
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
03.04.2018, 14:47
Ответы с готовыми решениями:

Spring, REST, Json
Добрый день, Возникает следующая ошибка: HTTP Status 415 - description The server refused this request because the request...

Spring, Rest, Json, LocalData
REST method POST вот такой json мапитса и все ок { &quot;mark&quot;:false, &quot;surname&quot;:&quot;test&quot;, &quot;name&quot;:&quot;test&quot;, ...

Ошибка формата при десериализации json
Json: Я так понимаю projects это массив Классы в C#

1
Эксперт Java
378 / 370 / 114
Регистрация: 30.06.2010
Сообщений: 1,445
04.04.2018, 05:38
Тебе же написали в ошибке что бы back reference нельзя использовать list<>(в твоем классе SubjectSection)

Вообще нельзя допускать выход управляемых(managed) объектов - сущностей jpa/hibernate вне транзакционного контекста, для этого делается отображение(маппинг) на dto объекты(тут погугли), т.е. методы сервиса возвращают dto. То же самое в обратном направлении. Принимаешь dto, передаешь в сервис, внутри сервиса мапишь на сущности и работаешь уже с доменной моделью.

Такой подход решает твой вопрос номер 2. В качестве маппера посмотри mapstruct
1
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
04.04.2018, 05:38
Помогаю со студенческими работами здесь

При десериализации JSON из файла происходит ошибка
using Newtonsoft.Json; using System; using System.Collections.Generic; using System.IO; using System.Runtime.Serialization; using...

Верификация файла .json и .xml при десериализации
Собственно проблема такая: нужно обработать импорт некорректного файла. Файлы содержат объекты базового класса собранные в...

Ошибка при десериализации json файла в list
Здравствуйте. У меня проблема с десериализации. Вот таким способом я делаю: var json = new...

Не определен конструктор без параметров. при десериализации json
Здравствуйте. Вот код Десерелизации. static void LoadJsonListInfoPrograms(string fileName) { ...

Ошибка при десериализации Json с Web.Api в Xamarin.Android
Получаю объекты класса UserInfo. После хочу чтобы метод вернул мне только FirstName. Но выбивают эту ошибку при десериализации Json. ...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
2
Ответ Создать тему
Опции темы

Новые блоги и статьи
Angular: Вопросы и ответы на собеседовании
Reangularity 15.06.2025
Готовишься к техническому интервью по Angular? Я собрал самые распространенные вопросы, с которыми сталкиваются разработчики на собеседованиях в этом году. От базовых концепций до продвинутых. . .
Архитектура Onion в ASP.NET Core MVC
stackOverflow 15.06.2025
Что такое эта "луковая" архитектура? Термин предложил Джеффри Палермо (Jeffrey Palermo) в 2008 году, и с тех пор подход только набирал обороты. Суть проста - представьте себе лук с его. . .
Unity 4D
GameUnited 13.06.2025
Четырехмерное пространство. . . Звучит как что-то из научной фантастики, правда? Однако для меня, как разработчика со стажем в игровой индустрии, четвертое измерение давно перестало быть абстракцией из. . .
SSE (Server-Sent Events) в ASP.NET Core и .NET 10
UnmanagedCoder 13.06.2025
Кажется, Microsoft снова подкинула нам интересную фичу в новой версии фреймворка. Работая с превью . NET 10, я наткнулся на нативную поддержку Server-Sent Events (SSE) в ASP. NET Core Minimal APIs. Эта. . .
С днём независимости России!
Hrethgir 13.06.2025
Решил побеседовать, с утра праздничного дня, с LM о завоеваниях. То что она написала о народе, представителем которого я являюсь сам сначала возмутило меня, но дальше только смешило. Это чисто. . .
Лето вокруг.
kumehtar 13.06.2025
Лето вокруг. Наполненное бурями и ураганами событий. На фоне магии Жизни, священной и вечной, неумелой рукой человека рисуется панорама душевного непокоя. Странные серые краски проникают и. . .
Популярные LM модели ориентированы на увеличение затрат ресурсов пользователями сгенерированного кода (грязь -заслуги чистоплюев).
Hrethgir 12.06.2025
Вообще обратил внимание, что они генерируют код (впрочем так-же ориентированы разработчики чипов даже), чтобы пользователь их использующий уходил в тот или иной убыток. Это достаточно опытные модели,. . .
Топ10 библиотек C для квантовых вычислений
bytestream 12.06.2025
Квантовые вычисления - это та область, где теория встречается с практикой на границе наших знаний о физике. Пока большая часть шума вокруг квантовых компьютеров крутится вокруг языков высокого уровня. . .
Dispose и Finalize в C#
stackOverflow 12.06.2025
Работая с C# больше десяти лет, я снова и снова наблюдаю одну и ту же историю: разработчики наивно полагаются на сборщик мусора, как на волшебную палочку, которая решит все проблемы с памятью. Да,. . .
Повышаем производительность игры на Unity 6 с GPU Resident Drawer
GameUnited 11.06.2025
Недавно копался в новых фичах Unity 6 и наткнулся на GPU Resident Drawer - штуку, которая заставила меня присвистнуть от удивления. По сути, это внутренний механизм рендеринга, который автоматически. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru