Sobes.tech
Назад к вопросам
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);