작성
·
63
0
학습하는 분들께 도움이 되고, 더 좋은 답변을 드릴 수 있도록 질문전에 다음을 꼭 확인해주세요.
1. 강의 내용과 관련된 질문을 남겨주세요.
2. 인프런의 질문 게시판과 자주 하는 질문(링크)을 먼저 확인해주세요.
(자주 하는 질문 링크: https://bit.ly/3fX6ygx)
3. 질문 잘하기 메뉴얼(링크)을 먼저 읽어주세요.
(질문 잘하기 메뉴얼 링크: https://bit.ly/2UfeqCG)
질문 시에는 위 내용은 삭제하고 다음 내용을 남겨주세요.
=========================================
[질문 템플릿]
1. 강의 내용과 관련된 질문인가요? (예)
2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? (예)
3. 질문 잘하기 메뉴얼을 읽어보셨나요? (예)
[질문 내용]
안녕하세요, 강의 잘 듣고 있습니다. 질문도 드리고, 맞다면 메뉴얼에 추가할 만한 내용이 있는 것 같아서 글 남겼습니다!
Page를 조회해올 때, left join을 사용한 경우, countQuery에 대해 명시하지 않더라도 count 쿼리문에서는 join 문이 확인되지 않았는데, 이것 또한 하이버네이트6에서 최적화하여 join 문을 생략한 것일까요?
하이버네이트6에서 left join을 직접 추가하더라도 내부적으로 최적화하는 질문을 보고 메뉴얼("스프링 부트 3 - 하이버네이트 6 left join 최적화 설명 추가")도 확인했습니다.
그럼에도 join을 강제로 수행시켜도 count 쿼리 또한 join 돼서 수행되는지 확인하고 싶어 다음과 같이 코드를 작성해봤습니다.
@Query(value = "select m,t from Member m left join m.team t")
Page<Member> findPageByAge(int age, Pageable pageable);
@Test
public void paging() {
//given
memberRepository.save(new Member("member1", 10));
memberRepository.save(new Member("member2", 10));
memberRepository.save(new Member("member3", 10));
memberRepository.save(new Member("member4", 10));
memberRepository.save(new Member("member5", 10));
int age = 10;
PageRequest pageRequest = PageRequest.of(0, 3, Sort.by(Direction.DESC, "username"));
Page<Member> page = memberRepository.findPageByAge(age, pageRequest);
//then
List<Member> members = page.getContent();
long totalElements = page.getTotalElements();
assertThat(members.size()).isEqualTo(3);
assertThat(totalElements).isEqualTo(5);
assertThat(page.getNumber()).isEqualTo(0);
assertThat(page.getTotalPages()).isEqualTo(2);
assertThat(page.isFirst()).isTrue();
assertThat(page.hasNext()).isTrue();
}
테스트 코드는 강의와 똑같이 작성하고, 쿼리문에서 select m, t
로 한 뒤 countQuery는 별도로 추가하지 않았습니다.
이때, 수행되는 쿼리가 다음과 같은데,
select
m1_0.member_id,
m1_0.age,
m1_0.team_id,
m1_0.username,
t1_0.team_id,
t1_0.name
from
member m1_0
left join
team t1_0
on t1_0.team_id=m1_0.team_id
order by
m1_0.username desc
fetch
first ? rows only
select
count(m1_0.member_id)
from
member m1_0
countQuery를 명시하지 않았음에도, 강의와 달리 count 쿼리문에서는 join이 수행되지 않았습니다.
답변 1
1
안녕하세요, 인프런 AI 인턴입니다. 천다인님.
질문해주신 count 쿼리에서 LEFT JOIN
이 수행되지 않는 이유에 대해 설명드리겠습니다. 스프링 데이터 JPA에서 페이징을 수행할 때, JPQL 쿼리에서 LEFT JOIN
을 명시한 경우에도, count 쿼리에서는 자동으로 JOIN이 생략됩니다. 이는 데이터베이스의 성능을 최적화하기 위한 전략 중 하나이며, 불필요한 JOIN을 제거하여 카운트 쿼리의 성능을 높이는 것이 기본 동작입니다.
이와 같은 최적화는 데이터베이스마다 다르게 동작할 수 있으며, 스프링 부트와 하이버네이트의 특정 버전에 따라서도 다를 수 있습니다. 따라서 특정 경우에는 직접 countQuery를 명시하여 JOIN을 포함시키도록 강제할 수 있습니다.
저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다. 현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏 추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.