인프런 커뮤니티 질문&답변

차재욱님의 프로필 이미지

작성한 질문수

실전! 스프링 부트와 JPA 활용1 - 웹 애플리케이션 개발

Fetch Join된 Table과 Fetch Join

23.10.20 15:32 작성

·

258

0

안녕하세요.

 

회사 신규 프로젝트로 Spring Boot와 JPA를 이용해 백엔드를 개발하고 있는 직장인입니다.

원래 C#개발자다 보니 수업을 들으며 차근차근 따라가 보지만 의외의 경우가 발생해 힘들지만

좋은 강의에 항상 재밌게 배우고 있습니다.

 

이번에 문의드릴 내용은 fetch join된 table에 다시 fetch join 하는 방법에 대해 문의 드리고자 합니다.

 

문제의 테이블 구조입니다.

Count Table과 Contract Table은 company_code라는 pk로 N:1구조

Contract Table은 Company Table과 company_code라는 pk로 N:1구조

구성돼 있습니다.

 

그리고 Count, Contract Table은 매핑되는 키에 대해서 FetchType을 Lazy로 설정한 상태입니다.

 

저는 여기서(JPQL기준) Count Table을 기준으로 Contract Table까지는 fetch join에 성공했지만 추가로 Company관련 쿼리가 N+1처럼 추가됐습니다.

 

혹시 제가 JPQL혹은 Query DSL방법으로 Contract Table과 Company Table을 fetch join 처리 후 다시 Count Table과 fetch join해 쿼리 하나로 해결 할 수 있는 방법이 있을까요?

 

감사합니다. :D

답변 2

0

김영한님의 프로필 이미지
김영한
지식공유자

2023. 10. 23. 20:17

안녕하세요. 차재욱님

활용2편에서 말씀드린 것 처럼

fetch join은 1:N의 경우 1번만 사용할 수 있습니다.

이런 경우 1:N 한번은 fetch join을 사용하고, 이후에는 batch fetch size와 같은 기술을 사용하셔야 합니다.

자세한 내용은 활용2편을 참고해주세요 :)

감사합니다.

0

인프런 AI 인턴님의 프로필 이미지

2023. 10. 22. 15:36

안녕하세요 차재욱 님, 인프런 AI 인턴이에요.

Fetch Join을 사용하여 한 번에 여러 테이블을 조회하는 것은 효율적인 방법입니다. 그러나 추가로 Join되는 테이블에 대해서도 N+1 문제가 발생할 수 있습니다.

이 경우에는 하위 테이블인 Contract Table과 Company Table을 한 번에 Fetch Join하여 해결할 수 있습니다. JPQL 또는 Query DSL을 사용하여 다음과 같이 쿼리를 작성할 수 있습니다.

JPQL 방법:

SELECT c FROM Count c
JOIN FETCH c.contract co
JOIN FETCH co.company

Query DSL 방법:

QCount count = QCount.count; // Count 엔티티에 대한 QueryDSL QType 변수
QContract contract = QContract.contract; // Contract 엔티티에 대한 QueryDSL QType 변수
QCompany company = QCompany.company; // Company 엔티티에 대한 QueryDSL QType 변수

List<Count> counts = queryFactory
    .selectFrom(count)
    .join(count.contract, contract).fetchJoin()
    .join(contract.company, company).fetchJoin()
    .fetch();

위의 쿼리를 사용하면 Count 테이블과 Contract 테이블, 그리고 그에 해당하는 Company 테이블까지 모두 한 번에 조회할 수 있습니다. 이렇게 함으로써 N+1 문제를 해결할 수 있습니다.

이러한 방법으로 Fetch Join을 통해 N+1 문제를 해결할 수 있습니다. 추가로 질문이 있으시면 언제든지 물어보세요. 감사합니다!