김영한님의 실전! querydsl 강의 내용을 정리한 노트입니다. 블로그에 있는 자료를 사용하실 때에는 꼭 김영한님 강의 링크를 남겨주세요!
public interface MemberRepository extends JpaRepository<Member, Long> {
//spring data jpa의 경우 save, findById, findAll을 제공함 -> 즉 만들 필요가 없음
//그러나 findByUsername은 제공하지 않음 -> 아래 처럼 만들면 됨, 메소드 이름으로 추측해서 알아서 쿼리를 쏴줌
List<Member> findByUsername(String username);
}
public interface MemberRepositoryCustom {
List<MemberTeamDto> search(MemberSearchCondition condition);
}
public class MemberRepositoryImpl implements MemberRepositoryCustom {
private final JPAQueryFactory queryFactory;
public MemberRepositoryImpl(EntityManager em) {
this.queryFactory = new JPAQueryFactory(em);
}
@Override
public List<MemberTeamDto> search(MemberSearchCondition condition) {
return queryFactory
.select(new QMemberTeamDto(
member.id.as("memberId"),
member.username,
member.age,
team.id.as("teamId"),
team.name.as("teamName")
))
.from(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(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 ageLoe(Integer ageLoe) {
return ageLoe != null ? member.age.loe(ageLoe) : null;
}
}
public interface MemberRepository extends JpaRepository<Member, Long>, MemberRepositoryCustom {
List<Member> findByUsername(String username);
}
@SpringBootTest
@Transactional
class MemberRepositoryTest {
@Autowired
EntityManager em;
@Autowired
MemberRepository memberRepository;
@Test
public void basicTest() {
Member member1 = new Member("member1", 10);
memberRepository.save(member1);
Member findMember = memberRepository.findById(member1.getId()).get();
assertThat(findMember).isEqualTo(member1);
List<Member> result1 = memberRepository.findAll();
assertThat(result1).containsExactly(member1);
List<Member> result2 = memberRepository.findByUsername("member1");
assertThat(result2).containsExactly(member1);
}
@Test
public void searchTest() {
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");
List<MemberTeamDto> result = memberRepository.search(condition);
for (MemberTeamDto memberTeamDto : result) {
System.out.println("memberTeamDto : " + memberTeamDto);
}
}
}
public interface MemberRepositoryCustom {
List<MemberTeamDto> search(MemberSearchCondition condition);
Page<MemberTeamDto> searchPageSimple(MemberSearchCondition condition, Pageable pageable);
Page<MemberTeamDto> searchPageComplex(MemberSearchCondition condition, Pageable pageable);
}
public class MemberRepositoryImpl implements MemberRepositoryCustom {
//..
@Override
public Page<MemberTeamDto> searchPageSimple(MemberSearchCondition condition, Pageable pageable) {
QueryResults<MemberTeamDto> results = queryFactory
.select(new QMemberTeamDto(
member.id.as("memberId"),
member.username,
member.age,
team.id.as("teamId"),
team.name.as("teamName")
))
.from(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(condition.getAgeLoe())
)
.offset(pageable.getOffset())
.limit(pageable.getPageSize())
.fetchResults();
List<MemberTeamDto> content = results.getResults();
long total = results.getTotal();
return new PageImpl<>(content, pageable, total);
}
//..
}
@Test
public void searchPageSimpleTest() {
//data 넣는 로직 생략
MemberSearchCondition condition = new MemberSearchCondition();
PageRequest pageRequest = PageRequest.of(0, 3);
Page<MemberTeamDto> result = memberRepository.searchPageSimple(condition, pageRequest);
assertThat(result.getSize()).isEqualTo(3);
assertThat(result.getContent()).extracting("username").containsExactly("member1", "member2", "member3");
}
public class MemberRepositoryImpl implements MemberRepositoryCustom {
//..
@Override
public Page<MemberTeamDto> searchPageComplex(MemberSearchCondition condition, Pageable pageable) {
//특정 조건에 걸리는 데이터와 전체 데이터 수를 조회할 때
List<MemberTeamDto> results = queryFactory
.select(new QMemberTeamDto(
member.id.as("memberId"),
member.username,
member.age,
team.id.as("teamId"),
team.name.as("teamName")
))
.from(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(condition.getAgeLoe())
)
.offset(pageable.getOffset())
.limit(pageable.getPageSize())
.fetch();
long total = queryFactory
.selectFrom(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(condition.getAgeLoe())
)
.fetchCount();
return new PageImpl<>(results, pageable, total);
}
//..
}
ctrl
+ alt
+ m
)@Override
public Page<MemberTeamDto> searchPageComplex(MemberSearchCondition condition, Pageable pageable) {
//특정 조건에 걸리는 데이터와 전체 데이터 수를 조회할 때
List<MemberTeamDto> results = getMemberTeamDtos(condition, pageable);
long total = getTotal(condition);
return new PageImpl<>(results, pageable, total);
}
@Override
public Page<MemberTeamDto> searchPageComplex(MemberSearchCondition condition, Pageable pageable) {
//특정 조건에 걸리는 데이터와 전체 데이터 수를 조회할 때
List<MemberTeamDto> results = queryFactory
.select(new QMemberTeamDto(
member.id.as("memberId"),
member.username,
member.age,
team.id.as("teamId"),
team.name.as("teamName")
))
.from(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(condition.getAgeLoe())
)
.offset(pageable.getOffset())
.limit(pageable.getPageSize())
.fetch();
JPAQuery<Member> countQuery = queryFactory
.selectFrom(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(condition.getAgeLoe())
);
return PageableExecutionUtils.getPage(results, pageable, () -> countQuery.fetchCount());
}
PageableExecutionUtils.getPage에 마지막 파라미터로 익명함수로 fetchCount()를 쏴주면 됨
그럼 이 getPage가 어떻게 생겼는지 한 번 보자!
@RestController
@RequiredArgsConstructor
public class MemberController {
private final MemberRepository memberRepository;
@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);
}
}
기본적으로 spring data가 Pageable 인터페이스를 넘기면 컨트롤러가 바인딩 될 때 데이터를 다 바인딩해서 넘겨줌 → 우리는 그걸 받아서 그대로 Page로 return 해주기만 하면 됨
이제 api 호출해보고 결과를 살펴보자! → localhost:8080/v2/members?page=0&size=5
→ 참고로 0 index이기 때문에 페이지 시작은 0페이지부터 시작, 지금 호출한 건 첫 페이지를 가리킴
아까 위 예제에서 만든 searchPageComplex를 호출해서 실제로 count 쿼리가 나가는지 안 나가는지 알아보자 → localhost:8080/v3/members?page=0&size=110
spring data를 사용하면 정렬 조건을 넣어줄 수 있음 → 그런데 springa data의 sort를 querydsl로 맞춰 쓰기가 굉장히 어려움. 그래서 spring data의 sort를 querydsl의 OrderSpecifier로 변환해서 사용하긴 해야함. 그런데 join 같은 복잡한 쿼리의 경우 OrderSpecifier가 의도한 방향으로 동작하지 않을 수 있음. 그렇기 때문에 sort를 파라미터로 받아서 직접 처리하는 것이 나을 수 있음.