Назад к вопросам
СБЕР Корпус
Junior — Middle
58
Объясните, почему при выполнении методов UserRepository.findById() и User.getEmails().forEach() может возникнуть проблема N+1, если метод getUsers() помечен как @Transactional, а поле email аннотировано как @OneToMany.
Компании, где спрашивали
Ответ от нейросети
sobes.tech AI
Проблема N+1 возникает из-за ленивой загрузки связей в JPA. В вашем случае:
- Метод
UserRepository.findById()загружает пользователя. - Поле
emailsпомечено как@OneToMany, по умолчанию загрузка связей — ленивое (FetchType.LAZY). - При вызове
User.getEmails().forEach()для каждого пользователя происходит отдельный запрос к базе, чтобы загрузить его email'ы.
Если метод getUsers() помечен как @Transactional, то сессия открыта и ленивые коллекции могут быть загружены при обращении. Но если вы перебираете список пользователей и для каждого вызываете getEmails(), то будет 1 запрос на пользователей + N запросов на email'ы — отсюда N+1.
Чтобы избежать этого, можно использовать:
JOIN FETCHв JPQL для предварительной загрузки коллекций.- Изменить стратегию загрузки на
EAGER(с осторожностью). - Использовать Entity Graphs.
Пример с JOIN FETCH:
@Query("SELECT u FROM User u JOIN FETCH u.emails WHERE u.id = :id")
User findByIdWithEmails(@Param("id") Long id);