Search

[Querydsl] 실무 활용 - 스프링 데이터 JPA와 Querydsl

상태
Querydsl
담당자
만든날짜
2022/05/17 11:47
수정날짜
2023/12/10 11:49
날짜
목차

스프링 데이터 JPA 리포지토리 변경

스프링 데이터 JPA - MemberRepository 생성
package study.querydsl.repository; import org.springframework.data.jpa.repository.JpaRepository; import study.querydsl.entitiy.Member; import java.util.List; public interface MemberRepository extends JpaRepository<Member, Long> { List<Member> findByUsername(String username); }
Java
복사
스프링 데이터 JPA 테스트
package study.querydsl.repository; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.transaction.annotation.Transactional; import study.querydsl.entitiy.Member; import javax.persistence.EntityManager; import java.util.List; import java.util.Optional; import static org.assertj.core.api.Assertions.assertThat; @SpringBootTest @Transactional class MemberRepositoryTest { @Autowired MemberRepository memberRepository; @Autowired EntityManager em; @Test public void basicTest() { Member member = new Member("member1", 10); memberRepository.save(member); Optional<Member> findMember = memberRepository.findById(member.getId()); assertThat(findMember.get()).isEqualTo(member); List<Member> result1 = memberRepository.findAll(); assertThat(result1).containsExactly(member); List<Member> result2 = memberRepository.findByUsername("member1"); assertThat(result2).containsExactly(member); } }
Java
복사
Querydsl 전용 기능인 회원 search를 작성할 수 없다. → 사용자 정의 리포지토리 필요

사용자 정의 리포지토리

사용자 정의 리포지토리 사용법
1.
사용자 정의 인터페이스 작성
2.
사용자 정의 인터페이스 구현
3.
스프링 데이터 리포지토리에 사용자 정의 이턴페이스 상속
사용자 정의 리포지토리 구성
1.
사용자 정의 인터페이스 작성
package study.querydsl.repository; import study.querydsl.dto.MemberSearchCondition; import study.querydsl.dto.MemberTeamDto; import java.util.List; public interface MemberRepositoryCustom { public List<MemberTeamDto> searchByWhere(MemberSearchCondition condition); }
Java
복사
2.
사용자 정의 인터페이스 구현
package study.querydsl.repository; import com.querydsl.core.types.dsl.BooleanExpression; import com.querydsl.jpa.impl.JPAQueryFactory; import lombok.RequiredArgsConstructor; import org.springframework.beans.factory.annotation.Autowired; import study.querydsl.dto.MemberSearchCondition; import study.querydsl.dto.MemberTeamDto; import study.querydsl.dto.QMemberTeamDto; import java.util.List; import static org.springframework.util.StringUtils.hasText; import static study.querydsl.entitiy.QMember.member; import static study.querydsl.entitiy.QTeam.team; @RequiredArgsConstructor public class MemberRepositoryImpl implements MemberRepositoryCustom { @Autowired private final JPAQueryFactory queryFactory; @Override public List<MemberTeamDto> searchByWhere(MemberSearchCondition condition) { return queryFactory .select(new QMemberTeamDto( member.id, member.username, member.age, team.id, team.name )).from(member) .leftJoin(member.team, team) .where( usernameEq(condition.getUsername()), teamNameEq(condition.getTeamName()), ageGoe(condition.getAgeGoe()), ageLog(condition.getAgeLoe()) ).fetch(); } private BooleanExpression usernameEq(String username) { return hasText(username) ? member.username.eq(username) : null; } private BooleanExpression teamNameEq(String teamName) { return hasText(teamName) ? team.name.eq(teamName) : null; } private BooleanExpression ageGoe(Integer ageGoe) { return ageGoe != null ? member.age.goe(ageGoe) : null; } private BooleanExpression ageLog(Integer ageLoe) { return ageLoe != null ? member.age.loe(ageLoe) : null; } }
Java
복사
3.
스프링 데이터 리포지토리에 사용자 정의 인터페이스 상속
package study.querydsl.repository; import org.springframework.data.jpa.repository.JpaRepository; import study.querydsl.entitiy.Member; import java.util.List; public interface MemberRepository extends JpaRepository<Member, Long>, MemberRepositoryCustom{ List<Member> findByUsername(String username); }
Java
복사
커스텀 리포지토리 동작 테스트 추가
@Test public void searchByWhereTest() throws Exception { //given Team teamA = new Team("teamA"); Team teamB = new Team("teamB"); em.persist(teamA); em.persist(teamB); Member member1 = new Member("member1", 10, teamA); Member member2 = new Member("member2", 20, teamA); Member member3 = new Member("member3", 30, teamB); Member member4 = new Member("member4", 40, teamB); em.persist(member1); em.persist(member2); em.persist(member3); em.persist(member4); MemberSearchCondition condition = new MemberSearchCondition(); condition.setAgeGoe(35); condition.setAgeLoe(40); condition.setTeamName("teamB"); //when List<MemberTeamDto> result = memberJpaRepository.searchByWhere(condition); //then assertThat(result).extracting("username").containsExactly("member4"); }
Java
복사

스프링 데이터 페이징 활용1 - Querydsl 페이징 연동

스프링 데이터의 Page, Pageable을 활용해보자.
전체 카운트를 한번에 조회하는 단순한 방법
데이터 내용과 전체 카운트를 별도로 조회하는 방법
사용자 정의 인터페이스 페이징 추가
package study.querydsl.repository; import org.springframework.data.domain.Page; import org.springframework.data.domain.Pageable; import study.querydsl.dto.MemberSearchCondition; import study.querydsl.dto.MemberTeamDto; import java.util.List; public interface MemberRepositoryCustom { public List<MemberTeamDto> searchByWhere(MemberSearchCondition condition); Page<MemberTeamDto> searchPageSimple(MemberSearchCondition condition, Pageable pageable); }
Java
복사
전체 카운트를 한번에 조회하는 방법
@Override public Page<MemberTeamDto> searchPageSimple(MemberSearchCondition condition, Pageable pageable) { List<MemberTeamDto> content = queryFactory .select(new QMemberTeamDto( member.id, member.username, member.age, team.id, team.name )).from(member) .leftJoin(member.team, team) .where( usernameEq(condition.getUsername()), teamNameEq(condition.getTeamName()), ageGoe(condition.getAgeGoe()), ageLog(condition.getAgeLoe()) ).offset(pageable.getOffset()) .limit(pageable.getPageSize()) .fetch(); return new PageImpl<>(content, pageable, content.size()); }
Java
복사
전체 카운트를 조회 하는 방법을 최적화 할 수 있으면 분리하는게 좋다.
코드를 리팩토링해서 내용 쿼리와 전체 카운트 쿼리를 읽기 좋게 분리하면 좋다.

스프링 데이터 페이징 활용2 - CountQuery 최적화

pagealbeExecutionUtils.getPage()로 최적화
@Override public Page<MemberTeamDto> searchPageComplex(MemberSearchCondition condition, Pageable pageable) { List<MemberTeamDto> content = queryFactory .select(new QMemberTeamDto( member.id, member.username, member.age, team.id, team.name )).from(member) .leftJoin(member.team, team) .where( usernameEq(condition.getUsername()), teamNameEq(condition.getTeamName()), ageGoe(condition.getAgeGoe()), ageLog(condition.getAgeLoe()) ).offset(pageable.getOffset()) .limit(pageable.getPageSize()) .fetch(); JPAQuery<MemberTeamDto> countQuery = queryFactory .select(new QMemberTeamDto( member.id, member.username, member.age, team.id, team.name )).from(member) .leftJoin(member.team, team) .where( usernameEq(condition.getUsername()), teamNameEq(condition.getTeamName()), ageGoe(condition.getAgeGoe()), ageLog(condition.getAgeLoe()) ); return PageableExecutionUtils.getPage(content, pageable, () -> countQuery.fetch().size()); }
Java
복사
스프링 데이터 라이브러리가 제공
count 쿼리가 생략 가능한 경우 생략해서 처리
페이지 시작이면서 컨텐츠 사이즈가 페이지 사이즈보다 작을 때
마지막 페이지 일 때 (offset + 컨텐츠 사이즈를 더해서 전체 사이즈 구함)

스프링 데이터 페이징 활용3 - 컨트롤러 개발

실제 컨트롤러
package study.querydsl.controller; import lombok.RequiredArgsConstructor; import org.springframework.data.domain.Page; import org.springframework.data.domain.Pageable; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RestController; import study.querydsl.dto.MemberSearchCondition; import study.querydsl.dto.MemberTeamDto; import study.querydsl.repository.MemberJpaRepository; import study.querydsl.repository.MemberRepository; import java.util.List; @RestController @RequiredArgsConstructor public class MemberController { private final MemberJpaRepository memberJpaRepository; private final MemberRepository memberRepository; @GetMapping("/v1/members") public List<MemberTeamDto> searchMemberV1(MemberSearchCondition condition) { return memberJpaRepository.searchByWhere(condition); } @GetMapping("/v2/members") public Page<MemberTeamDto> searchMemberV2(MemberSearchCondition condition, Pageable pageable) { return memberRepository.searchPageSimple(condition, pageable); } @GetMapping("/v3/members") public Page<MemberTeamDto> searchMemberV3(MemberSearchCondition condition, Pageable pageable) { return memberRepository.searchPageComplex(condition, pageable); } }
Java
복사
스프링 데이터 Sort를 Querydsl의 OrderSpecifier로 변환
JPAQuery<Member> query = queryFactory .selectFrom(member); for (Sort.Order o : pageable.getSort()) { PathBuilder pathBuilder = new PathBuilder(member.getType(), member.getMetadata()); query.orderBy(new OrderSpecifier(o.isAscending() ? Order.ASC : Order.DESC, pathBuilder.get(o.getProperty()))); } List<Member> result = query.fetch();
Java
복사
참고: 정렬 Sort 은 조건이 조금만 복잡해져도 Pageable 의 Sort 기능을 사용하기 어렵다. 루트 엔티티 범위를 넘어가는 동적 정렬 기능이 필요하면 스프링 데이터 페이징이 제공하는 Sort를 사용하기 보다는 파라미터를 받아서 직접 처리하는 것을 권장한다.