이번 시간에는 간단하게 순수 JPA로 만든 코드를 스프링 데이터 JPA로 바꿔보겠다.
✔️ MemberRepository.java
→ 인터페이스
public interface MemberRepository extends JpaRepository<Member, Long> {
// select m from Member m where m.username = ?
List<Member> findByUsername(String username);
}
테스트 파일 생성
✔️ MemberRepositoryTest.java
이전에 작성했던 코드를 가져와서 단축키를 이용해 한 번에 바꿔줌
@SpringBootTest
@Transactional
public class MemberRepositoryTest {
@Autowired
EntityManager em;
@Autowired MemberRepository memberRepository; // shift + fn + F6으로 변수명을 한꺼번에 바꿀 수 있음
@Test
public void basicTest() {
Member member = new Member("member1", 10);
memberRepository.save(member);
Member findMember = memberRepository.findById(member.getId()).get();
assertThat(findMember).isEqualTo(member);
List<Member> result1 = memberRepository.findAll();
assertThat(result1).containsExactly(member);
List<Member> result2 = memberRepository.findByUsername("member1");
assertThat(result2).containsExactly(member);
}
}
단축키
shift + fn + F6
: 변수명 한꺼번에 바꾸기
(querydsl을 쓰려면 결국 구현 코드를 만들어야 함.
스프링 데이터는 인터페이스로 동작함.)
📌 사용자 정의 리포지토리 사용법
- 사용자 정의 인터페이스 작성
- 사용자 정의 인터페이스 구현
- 스프링 데이터 리포지토리에 사용자 정의 인터페이스 상속
public interface MemberRepositoryCustom {
List<MemberTeamDto> search(MemberSearchCondition condition);
}
// MemberRepository + Impl 이라는 이름으로 만들어줘야 함
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")
))
// .selectFrom(member) // (장점) 이렇게 select projection이 달라져도 where절 안에 있는 것들을 재사용할 수 있음
.from(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(condition.getAgeLoe())
)
.fetch();
}
// usernameEq, teamNameEq, ageGoe, ageLoe는 재사용 가능
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 {
// select m from Member m where m.username = ?
List<Member> findByUsername(String username);
}
다 됐으면 예전에 사용했던 searchTest
의 내용을 가져와 memberRepository로만 수정해 테스트해보면 된다.
Page
, Pageable
을 활용할 것✔️ MemberRepositoryCustom.java
에 아래의 코드 추가
Page<MemberTeamDto> searchPageSimple(MemberSearchCondition condition, Pageable pageable);
Page<MemberTeamDto> searchPageComplex(MemberSearchCondition condition, Pageable pageable);
✔️ MemberRepositoryImpl.java
@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(); // fetchReesults: querydsl이 count 쿼리랑 content 쿼리 이렇게 두 번 날림
List<MemberTeamDto> content = results.getResults();
long total = results.getTotal();
return new PageImpl<>(content, pageable, total);
}
➡️ searchPageSimple()
, fetchResults()
사용
fetchResults()
는 내용과 전체 카운트를 한 번에 조회할 수 있다. (실제 쿼리 2번 호출됨)fetchResults()
는 카운트 쿼리 실행시 필요없는 order by
는 제거한다. @Override
public Page<MemberTeamDto> searchPageComplex(MemberSearchCondition condition, Pageable pageable) {
List<MemberTeamDto> content = 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();
// 내가 직접 total count 쿼리를 날리는 것
long total = queryFactory
.select(member)
.from(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(condition.getAgeLoe())
)
.fetchCount();
return new PageImpl<>(content, pageable, total);
}
➡️ searchPageComplex()
사용
/**
* 바로 위의 코드에서 CountQuery 최적화하기
* PageableExecutionUtils.getPage()로 최적화
*/
JPAQuery<Member> countQuery = queryFactory
.select(member)
.from(member)
.leftJoin(member.team, team)
.where(
usernameEq(condition.getUsername()),
teamNameEq(condition.getTeamName()),
ageGoe(condition.getAgeGoe()),
ageLoe(condition.getAgeLoe())
);
return PageableExecutionUtils.getPage(content, pageable, countQuery::fetchCount);
➡️ PageableExecutionUtils.getPage()
로 최적화 할 수 있다.
✔️ MemberController.java
@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);
➡️ page=1로 바꾸면 다음 페이지로 (제로 인덱스라 0부터 시작함)
스프링 데이터 JPA는 자신의 정렬(Sort)을 Querydsl의 정렬 (OrderSpecifier)로 편리하게 변경하는 기능을 제공한다.
(이 부분은 뒤에서 다시 말씀하신다고 하심)
⚠️ 정렬(sort)은 조건이 조금만 복잡해져도 Pageable의 Sort 기능을 사용하기 어렵다.
루트 엔티티 범위를 넘어가는 동적 정렬 기능이 필요하면 스프링 데이터 페이징이 제공하는 sort를 사용하기 보다는 파라미터를 받아 직접 처리하는 것을 권장한다.