채널톡 아이콘

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

천다인님의 프로필 이미지

작성한 질문수 4

실전! 스프링 데이터 JPA

스프링 데이터 JPA 페이징과 정렬

왜 select에서 left join을 수행하더라도 count에서는 left join을 수행하지 않나요?

작성

·

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 인턴님의 프로필 이미지

안녕하세요, 인프런 AI 인턴입니다. 천다인님.

질문해주신 count 쿼리에서 LEFT JOIN이 수행되지 않는 이유에 대해 설명드리겠습니다. 스프링 데이터 JPA에서 페이징을 수행할 때, JPQL 쿼리에서 LEFT JOIN을 명시한 경우에도, count 쿼리에서는 자동으로 JOIN이 생략됩니다. 이는 데이터베이스의 성능을 최적화하기 위한 전략 중 하나이며, 불필요한 JOIN을 제거하여 카운트 쿼리의 성능을 높이는 것이 기본 동작입니다.

핵심 개념 정리

  • SELECT 쿼리에 대한 LEFT JOIN: 데이터 자체를 조회할 때는 필요에 따라 LEFT JOIN을 사용해 연관된 데이터를 함께 가져옵니다.
  • COUNT 쿼리에 대한 JOIN 제거: 스프링 데이터 JPA는 JOIN을 꼭 필요로 하지 않는 한, COUNT 쿼리에서는 JOIN을 제거하여 쿼리 성능을 개선합니다. COUNT 쿼리는 오직 레코드 수를 세기 위한 것이므로, JOIN 없이 더 빠르게 수행될 수 있습니다.

참고할 수 있는 유사 질문

이와 같은 최적화는 데이터베이스마다 다르게 동작할 수 있으며, 스프링 부트와 하이버네이트의 특정 버전에 따라서도 다를 수 있습니다. 따라서 특정 경우에는 직접 countQuery를 명시하여 JOIN을 포함시키도록 강제할 수 있습니다.

저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다. 현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏 추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.