일대일 관계에서 N+1 문제
안녕하세요 강사님 강의 보면서 많이 배우고 적용해보고있습니다.
테스트를 해보던중 이해할수 없는 추가적인 쿼리가 발생해서 질문드립니다.
먼저 예약과 리뷰 엔티티 클래스입니다.
@Entity
@Getter
public class Reservation extends BaseEntity {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "reservation_id")
private Long id;
...
@OneToOne(mappedBy = "reservation")
private Review review;
}
@Entity
@Getter
@Table(name = "rental_home_review")
public class Review extends BaseEntity {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "rental_home_review_id")
private Long id;
@OneToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "reservation_id", nullable = false)
private Reservation reservation;
private int score;
private String content;
@Enumerated(EnumType.STRING)
private WritingStatus status;
}
@PersistenceUnit
EntityManagerFactory emf;
@Test
public void 조인_jpql() {
String query = "select rh from RentalHome rh join rh.reservations where rh.id = 6486L";
RentalHome rentalHome = em.createQuery(query, RentalHome.class).getSingleResult();
boolean rentalHomeLoaded = emf.getPersistenceUnitUtil().isLoaded(rentalHome);
boolean reservationsLoaded = emf.getPersistenceUnitUtil().isLoaded(rentalHome.getReservations());
assertThat(rentalHomeLoaded).isTrue();
assertThat(reservationsLoaded).isFalse();
}
@Test
public void 패치조인_jpql() {
String query = "select rh from RentalHome rh join fetch rh.reservations where rh.id = 6486L";
RentalHome rentalHome = em.createQuery(query, RentalHome.class).getSingleResult();
boolean rentalHomeLoaded = emf.getPersistenceUnitUtil().isLoaded(rentalHome);
boolean reservationsLoaded = emf.getPersistenceUnitUtil().isLoaded(rentalHome.getReservations());
assertThat(rentalHomeLoaded).isTrue();
assertThat(reservationsLoaded).isTrue();
}위의 테스트 메서드에서 조인_jpql 쿼리는 의도한대로 테스트 통과가 맞고
패치조인_jpql에서도 테스트는 통과하지만 추가적인 쿼리가 발생합니다.

Reservation과 Review의 관계를 Lazy로 설정했고
Reivew 객체는 사용하지도 않았는데 추가적인 쿼리가 왜 발생했는지 모르겠어서 질문 올립니다.
답변 1
0
안녕하세요, 인프런 AI 인턴입니다. wnsdus1008님.
일대일 양방향 연관관계에서 N+1 문제가 발생하는 이유는 주로 연관된 엔티티들이 조회될 때 프록시 초기화가 발생하기 때문입니다. 특히, @OneToOne 관계에서 지연 로딩을 설정해도 관계된 엔티티의 데이터를 실제로 접근하게 될 때 프록시가 초기화되며 추가적인 쿼리가 발생할 수 있습니다.
비슷한 문제가 논의된 글들을 참고해보시면 좋겠습니다:
특히, 두 번째 URL에서 다루는 예시는 여러 엔티티와 프록시 객체의 초기화 문제를 잘 설명하고 있습니다.
주요 포인트 요약:
Lazy Loading의 한계:
@OneToOne또는@OneToMany등의 연관관계에서 지연 로딩을 설정해도 프록시 객체가 초기화되며 추가 쿼리가 발생할 수 있습니다.Fetch Join 사용 시 주의점: 조인이나 페치 조인을 사용할 때, 관련된 모든 엔티티를 한 번에 가져오게 되며, 지연 로딩과 상관없이 필요 시 추가 쿼리가 발생할 수 있습니다.
N + 1 문제 해결: 문제를 해결하기 위해 페치 조인을 사용할 수 있지만, 적절한 페이징이나 쿼리 튜닝이 필수적입니다. 페치 조인을 사용할 때는 데이터 양의 폭증을 주의해야 합니다.
아래 코드 실행 시에도 지연 로딩이 발생하는 것을 확인할 수 있습니다.
@Test
@Transactional
void test() {
Review review = new Review();
review.setContent("This is a review");
reviewRepository.save(review);
Reservation reservation = new Reservation();
reservation.setReview(review);
reservationRepository.save(reservation);
entityManager.flush();
entityManager.clear();
// ---------------------//
Reservation findReservation = reservationRepository.findById(1L).get(); // Reservation 엔티티 조회
System.out.println("findReservation.getReview() start");
Review findReview = findReservation.getReview();
System.out.println("findReservation.getReview() end");
System.out.println("findReview.getContent() start");
String content = findReview.getContent(); // Review 엔티티 지연 로딩
System.out.println("findReview.getContent() end");
}
실제 실행 결과를 보면, 지연 로딩 설정에도 불구하고 Review에 접근할 때 추가적인 쿼리가 발생할 수 있습니다.
좀 더 상세한 해결책이나 최적화 방법에 대해 알고 싶다면,
저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다.
현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏
추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.
강의 관련 외 질문입니다.
0
81
2
SpringBoot4 + Hibernate7 모듈 등록 방법 공유
1
99
1
BeanCreationException
0
96
3
Update 후 UpdateMemberResponse 매핑할 때
0
57
1
트랜잭션을 사용 안 할 때 커넥션은 언제 가져오나요?
0
102
2
페이징 + 검색조건 관련해서 질문드립니다.
0
74
1
Query Dsl Q파일 질문입니다.
0
86
1
루트 쿼리라는것은
0
62
1
메서드를 분리하는 기준
0
71
1
findAllWithMemberDelivery 메서드 질문드립니다.
0
116
3
연관관계 매핑을 안 쓸 경우, 사용해야 하는 전략
0
92
2
fetch join과 영속화와 OSIV의 관계
0
90
2
Distinct 사용 전 결과에 대한 의문
0
118
2
레포지토리 계층에서의 트랜잭션에 대한 의문
0
63
1
영속성 컨텍스트 생명주기의 신기한 부분이 있습니다.
0
79
2
dto 필드 속 엔티티 여부
0
63
1
뷰템플릿 사용 시
0
82
2
Result 클래스 관련 질문
0
56
1
@PostConstruct 프록시 관련 질문드립니다
0
88
1
DTO 대신 Form 사용은 안되나요?
0
140
1
OSIV ON 상태일 때
0
99
1
fetch join VS fetch join 페이징 궁금증
0
190
2
양방향 연관관계 알아보는 법?
0
110
1
16강 17강 간단 정리 이게 맞을까요 ?
0
168
2





