Форум программистов, компьютерный форум, киберфорум
Java EE (J2EE)
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.94/18: Рейтинг темы: голосов - 18, средняя оценка - 4.94
6 / 6 / 1
Регистрация: 04.01.2017
Сообщений: 465

Рекурсия в Response JSON

21.03.2021, 21:34. Показов 3946. Ответов 8

Студворк — интернет-сервис помощи студентам
Здравствуйту!

Подскажите как предотвратить вот такую рекурсию в ответах JSON на запрос:
JSON
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
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
{
    "idUser": 8,
    "userName": "vldmr3",
    "password": "$2a$10$EyqmdDtDup5FXQ0Ut.Se1eSVRnMieuPTOrEWBhSB8HBDUYsYcgNdu",
    "confirmPassword": null,
    "wordCoder": "password",
    "status": "ACTIVE",
    "email": "vldmr1@gmail.com",
    "firstName": "Volodya",
    "lastName": "Vladimir",
    "createdDate": 1616350626000,
    "updatedDate": null,
    "requests": [],
    "messages": [],
    "userData": {
        "idRUserData": 8,
        "fullNameCompany": "Unknown",
        "user": {
            "idUser": 8,
            "userName": "vldmr3",
            "password": "$2a$10$EyqmdDtDup5FXQ0Ut.Se1eSVRnMieuPTOrEWBhSB8HBDUYsYcgNdu",
            "confirmPassword": null,
            "wordCoder": "password",
            "status": "ACTIVE",
            "email": "vldmr1@gmail.com",
            "firstName": "Volodya",
            "lastName": "Vladimir",
            "createdDate": 1616350626000,
            "updatedDate": null,
            "requests": [],
            "messages": [],
            "userData": {
                "idRUserData": 8,
                "fullNameCompany": "Unknown",
                "user": {
                    "idUser": 8,
                    "userName": "vldmr3",
                    "password": "$2a$10$EyqmdDtDup5FXQ0Ut.Se1eSVRnMieuPTOrEWBhSB8HBDUYsYcgNdu",
                    "confirmPassword": null,
                    "wordCoder": "password",
                    "status": "ACTIVE",
                    "email": "vldmr1@gmail.com",
                    "firstName": "Volodya",
                    "lastName": "Vladimir",
                    "createdDate": 1616350626000,
                    "updatedDate": null,
                    "requests": [],
                    "messages": [],
                    "userData": {
                        "idRUserData": 8,
                        "fullNameCompany": "Unknown",
                        "user": {
                            "idUser": 8,
                            "userName": "vldmr3",
                            "password": "$2a$10$EyqmdDtDup5FXQ0Ut.Se1eSVRnMieuPTOrEWBhSB8HBDUYsYcgNdu",
                            "confirmPassword": null,
                            "wordCoder": "password",
                            "status": "ACTIVE",
                            "email": "vldmr1@gmail.com",
                            "firstName": "Volodya",
                            "lastName": "Vladimir",
                            "createdDate": 1616350626000,
                            "updatedDate": null,
                            "requests": [],
                            "messages": [],
                            "userData": {
                                "idRUserData": 8,
                                "fullNameCompany": "Unknown",
                                "user": {
                                    "idUser": 8,
                                    "userName": "vldmr3",
                                    "password": "$2a$10$EyqmdDtDup5FXQ0Ut.Se1eSVRnMieuPTOrEWBhSB8HBDUYsYcgNdu",
                                    "confirmPassword": null,
                                    "wordCoder": "password",
                                    "status": "ACTIVE",
                                    "email": "vldmr1@gmail.com",
                                    "firstName": "Volodya",
                                    "lastName": "Vladimir",
                                    "createdDate": 1616350626000,
                                    "updatedDate": null,
                                    "requests": [],
                                    "messages": [],
                                    "userData": {
Методы toString() в классах User and UserData переопределил. но это не помогло:
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
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
@Entity
@Table(name="Users")
public class User {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name="Id_user")
    private long idUser;
    @Column(name="User_name")
    private  String userName;
    @Column(name="Password")
    private String password;
    @Transient
    transient private String confirmPassword;
    @Column(name="Word_Coder")
    private String wordCoder;
    @Enumerated(value = EnumType.STRING)
    @Column(name="Status")
    private Status status;
    @Column(name="Email")
    private String email;
    @Column(name="First_name")
    private String firstName;
    @Column(name="Last_name")
    private String lastName;
    @Column(name="Created_date")
    private Date createdDate;
    @Column(name="Updated_date")
    private Date updatedDate;
 
    @OneToMany(mappedBy="userRequest", fetch = FetchType.EAGER)
    private List<Request> requests;
 
    @OneToMany(mappedBy="userResponse")
    private List<Message> messages;
 
    @OneToOne(cascade = CascadeType.ALL,fetch = FetchType.EAGER)
    @JoinColumn(name = "Data_user_id", referencedColumnName = "Id_data_user")
    private UserData userData;
 
    @ManyToMany(cascade = {CascadeType.ALL})
    @JoinTable(name = "User_roles", joinColumns = {@JoinColumn(name="User_id")},
    inverseJoinColumns = {@JoinColumn(name = "Role_id")})
    private Set<Role> roles;
 
    public User(long id_User, String firstName, String lastName) {
        this.idUser = id_User;
        this.firstName = firstName;
        this.lastName = lastName;
    }
 
    @Override
    public String toString() {
        return "User{" +
                "login='" + userName + '\'' +
                ", password='" + password + '\'' +
                ", wordCoder='" + wordCoder + '\'' +
                ", status=" + status.toString() +
                ", email='" + email + '\'' +
                ", firstName='" + firstName + '\'' +
                ", lastName='" + lastName + '\'' +
                ", createdDate=" + createdDate.toString() +
                ", updatedDate=" + updatedDate.toString() +
                ", requests=" + requests.toString() +
                ", messages=" + messages.toString() +
                ", userData=" + userData.getFullNameCompany()+" "+userData.getINN() +
                ", countRoles=" + roles.size() +
                '}';
    }
}
 
@Entity
@Table(name="data_user")
public class UserData {
 
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name = "Id_data_user")
    private Long idRUserData;
 
    @Column(name="INN")
    private  String INN;
 
    @Column(name="Full_name_company")
    private  String fullNameCompany;
 
    @OneToOne(mappedBy = "userData")
    private User user;
 
    @ManyToOne
    @JoinColumn(name = "id_type_subject")
    private TypeSubject typeSubject;
 
    @ManyToOne
    @JoinColumn(name = "id_organizational_form")
    private OrganizationalForm organizationalForm;
 
    @Override
    public String toString() {
        return "UserData{" +
                "INN='" + INN + '\'' +
                ", fullNameCompany='" + fullNameCompany + '\'' +
                ", user=" + user.getLogin().toString() +
                ", typeSubject=" + typeSubject.toString() +
                ", organizationalForm=" + organizationalForm.toString() +
                '}';
    }
}
Получение объекта из БД:
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
31
32
33
34
35
36
37
38
39
40
41
42
 @GetMapping(path = "/user/{id}", produces = "application/json")
    public User getById(@PathVariable("id") long id){
 
        return userService.findById(id).get();
    }
 
@Service
public class UserServiceImpl implements UserService {
 
    @Override
    public Optional<User> findById(Long id) {
        return userRepository.findById(id);
    }
 
    @Override
    public Optional<User> findByUserName(String login) {
        return userRepository.findByUserName(login);
    }
 
    @Override
    public List<User> findAllBy() {
        return userRepository.findAllBy();
    }
 
    @Override
    public List<User> findByUserData(UserData userData) {
        return userRepository.findByUserData(userData);
    }
}
 
@Repository
public interface UserRepository extends JpaRepository<User,Long>,UserCustomRepository {
 
    Optional<User> findById(Long id);
    Optional<User> findByUserName(String userName);
    List<User> findAllBy();
    List<User> findByUserData(UserData userData);
 
    //@Query(value = "SELECT * FROM users WHERE Status=:status",nativeQuery = true)
    List<User> findByStatus(@Param("status")Status status);
 
}
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
21.03.2021, 21:34
Ответы с готовыми решениями:

Не могу разобраться с тем, как парсить json: response.json as?
не могу разобраться с тем как парсить json response.json as? предполагаю что проблема в том что я неправильно составляю то что...

JSON запрос (некорректный response)
После запроса id 1194 возврат output с товаром id 1194, id 1195 возврат output с товаром id 1194, (должен быть...

Json.net and vk api response object
response: Все перепробовал

8
Эксперт Java
3639 / 2971 / 918
Регистрация: 05.07.2013
Сообщений: 14,220
21.03.2021, 22:14
Лучший ответ Сообщение было отмечено Vlad__i__mir как решение

Решение

Не надо отдавать ентити из контроллера
1
6 / 6 / 1
Регистрация: 04.01.2017
Сообщений: 465
21.03.2021, 22:33  [ТС]
Цитата Сообщение от xoraxax Посмотреть сообщение
Не надо отдавать ентити из контроллера
Это плохая практика или это связано конкретно с данной ситуацией?

Почему спрашиваю, потому что у меня там у объекта есть коллекции и их придется как-то обрабатывать
0
Эксперт Java
3639 / 2971 / 918
Регистрация: 05.07.2013
Сообщений: 14,220
21.03.2021, 23:12
Обрабатывай
0
2 / 1 / 1
Регистрация: 17.08.2020
Сообщений: 3
31.03.2021, 11:04
Лучший ответ Сообщение было отмечено Vlad__i__mir как решение

Решение

Вообще в библиотеке Jackson (если вы её используете) есть специальные аннотации, которые специально для решения этой проблемы предназначены.

@JsonBackReference и @JsonManagedReference

Они как раз нужны для того, чтобы определить родительский и дочерний объект и убрать рекурсию. Почитайте вот здесь.

Добавлено через 2 минуты
И, кстати говоря, не знаю такого правила, что ни в коем случае нельзя отдавать сущность из контроллера. Всё от ситуации зависит.

А почему собственно нет? Лишние поля, которые в json не нужны, можно отметить @JsonIgnore и получится на выходе милый json. Для маленького проекта делать TO может быть вообще нецелесообразно (особенно если поля там будут дублировать поля в сущности)
1
Эксперт Java
3639 / 2971 / 918
Регистрация: 05.07.2013
Сообщений: 14,220
31.03.2021, 11:15
Eugenue, ага, еще есть @JsonView например. Допустим, они работают всегда так, как ожидается. Но в общем случае то, что хранится и то, что надо показывать пользователю - это разные вещи, одно может изменяться независимо от другого. Это самая очевидная причина, можно еще немного углубиться при желании.
0
2 / 1 / 1
Регистрация: 17.08.2020
Сообщений: 3
31.03.2021, 11:22
да я не спорю. Но технических проблем это не создаст, если так делать (если модель меняться не будет, конечно, иначе создаст). И иногда, имхо, лучше так и делать (для петов или быстрых проектов).
0
Эксперт Java
3639 / 2971 / 918
Регистрация: 05.07.2013
Сообщений: 14,220
31.03.2021, 11:36
ну для себя ты конечно можешь делать все, что угодно, кто ж тебе запретит
0
6 / 6 / 1
Регистрация: 04.01.2017
Сообщений: 465
06.04.2021, 19:58  [ТС]
xoraxax, Eugenue,

Понял, спасибо
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
06.04.2021, 19:58
Помогаю со студенческими работами здесь

Корректность json для retrofit Response
Есть Shipment.json такого содержания: private int id; private int pickupDateFrom; private int pickupDateTo; private Map&lt;Integer,...

Свойства Response.Expires = 60 Response.Expiresabsolute = Now() - 1 Response.AddHeader 'pragma','no-cache' Response.AddHeader 'cache-control','p
Нашел в Инете строчки для запрета буфиризации и кэширования страници бравзером Response.Expires = 60 Response.Expiresabsolute = Now() -...

ResponseBodyAdvice, обертка для Json response в Spring
Здравствуйте, нужно в json получить данные вида: { &quot;data&quot;: { &quot;result&quot;:&quot;success&quot; } } для этих целей есть класс - обертка...

Laravel 4.2 Возврат отрендериного шаблона через Response::json()
Доброго времение суток, уважаемые форумчане. Столкнулся с проблемой. На стороне клиента посылается ajax запрос. ...

Разбор JSON - рекурсия vs итерация
есть функция для анализа JSON, выдает список всех ключей d = , &quot;cars&quot;: {&quot;model&quot;: &quot;Лада Калина&quot;, &quot;mpg&quot;: 37.5}, },...


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

Или воспользуйтесь поиском по форуму:
9
Ответ Создать тему
Новые блоги и статьи
SDL3 для Web (WebAssembly): Вывод текста со шрифтом TTF с помощью SDL3_ttf
8Observer8 01.02.2026
Содержание блога В этой пошаговой инструкции создадим с нуля веб-приложение, которое выводит текст в окне браузера. Запустим на Android на локальном сервере. Загрузим Release на бесплатный. . .
SDL3 для Web (WebAssembly): Сборка C/C++ проекта из консоли
8Observer8 30.01.2026
Содержание блога Если вы откроете примеры для начинающих на официальном репозитории SDL3 в папке: examples, то вы увидите, что все примеры используют следующие четыре обязательные функции, а. . .
SDL3 для Web (WebAssembly): Установка Emscripten SDK (emsdk) и CMake для сборки C и C++ приложений в Wasm
8Observer8 30.01.2026
Содержание блога Для того чтобы скачать Emscripten SDK (emsdk) необходимо сначало скачать и уставить Git: Install for Windows. Следуйте стандартной процедуре установки Git через установщик. . . .
SDL3 для Android: Подключение Box2D v3, физика и отрисовка коллайдеров
8Observer8 29.01.2026
Содержание блога Box2D - это библиотека для 2D физики для анимаций и игр. С её помощью можно определять были ли коллизии между конкретными объектами. Версия v3 была полностью переписана на Си, в. . .
Инструменты COM: Сохранение данный из VARIANT в файл и загрузка из файла в VARIANT
bedvit 28.01.2026
Сохранение базовых типов COM и массивов (одномерных или двухмерных) любой вложенности (деревья) в файл, с возможностью выбора алгоритмов сжатия и шифрования. Часть библиотеки BedvitCOM Использованы. . .
SDL3 для Android: Загрузка PNG с альфа-каналом с помощью SDL_LoadPNG (без SDL3_image)
8Observer8 28.01.2026
Содержание блога SDL3 имеет собственные средства для загрузки и отображения PNG-файлов с альфа-каналом и базовой работы с ними. В этой инструкции используется функция SDL_LoadPNG(), которая. . .
SDL3 для Android: Загрузка PNG с альфа-каналом с помощью SDL3_image
8Observer8 27.01.2026
Содержание блога SDL3_image - это библиотека для загрузки и работы с изображениями. Эта пошаговая инструкция покажет, как загрузить и вывести на экран смартфона картинку с альфа-каналом, то есть с. . .
Влияние грибов на сукцессию
anaschu 26.01.2026
Бифуркационные изменения массы гриба происходят тогда, когда мы уменьшаем массу компоста в 10 раз, а скорость прироста биомассы уменьшаем в три раза. Скорость прироста биомассы может уменьшаться за. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru