Notice
Recent Posts
Recent Comments
Link
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
Tags
- db
- transaction
- 그리디
- 자바
- 김영한
- spring
- JDBC
- JPQL
- 인프런
- 스프링 핵심 원리
- 백준
- 스프링
- Spring Boot
- 알고리즘
- QueryDSL
- Android
- Greedy
- 스프링 핵심 기능
- java
- Exception
- Servlet
- kotlin
- springdatajpa
- pointcut
- jpa
- AOP
- Proxy
- http
- Thymeleaf
- SpringBoot
Archives
- Today
- Total
개발자되기 프로젝트
SpringDataJpa 페이징, 정렬 본문
1. SpringDataJpa
스프링에서 페이징 기능을 표준화 시켰다 ㄷㄷㄷ
2. 페이징, 정렬 파라미터
- org.springframework.data.domain.Sort : 정렬 기능
- org.springframework.data.domain.Pageable : 페이징 기능 (내부에 Sort 포함)
- data.jpa가 아니다 즉, DB에 관계없이 적용이 가능한다.
3. 특별한 반환 타입
- org.springframework.data.domain.Page : 추가 count 쿼리 결과를 포함하는 페이징
- TotalCount 가져오는 쿼리는 자동으로 날라감.
- org.springframework.data.domain.Slice : 추가 count 쿼리 없이 다음 페이지만 확인 가능
(내부적으로 limit + 1조회) --> limit이 10개면 11개 조회해둠 ㅋㅋ - List (자바 컬렉션): 추가 count 쿼리 없이 결과만 반환
4. 예제
- 조건
- 검색 조건: 나이가 10살
- 정렬 조건: 이름으로 내림차순
- 페이징 조건: 첫 번째 페이지, 페이지당 보여줄 데이터는 3건
- Repository
- Page를 반환 타입으로 받고
- 입력 값으로 Pageable이 필요하다.
- Pageable : Paging informain의 추상 인터페이스(구현체는 PageRequest)
Page<Member> findByAge(int age, Pageable pageable);
- Test
- Paging처리를 위해 PageRequest에 paging정보를 담아서 건내줘야 한다.
- page는 다양한 기능을 제공한다.
- .getTotalElement(), .getNumber() 등등
@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(Sort.Direction.DESC, "userName"));
//when
Page<Member> page = memberRepository.findByAge(age, pageRequest);
//then
List<Member> content = page.getContent(); //data 꺼내기
long totalCount = page.getTotalElements(); //totalCount
Assertions.assertThat(content.size()).isEqualTo(3);
Assertions.assertThat(page.getTotalElements()).isEqualTo(5);
Assertions.assertThat(page.getNumber()).isEqualTo(0); //page 번호
Assertions.assertThat(page.getTotalPages()).isEqualTo(2);
Assertions.assertThat(page.isFirst()).isTrue();
Assertions.assertThat(page.hasNext()).isTrue();
}
- 결과 : return type이 Page이기 때문에 count호출하는 쿼리가 나감.
select
member0_.member_id as member_i1_0_,
member0_.age as age2_0_,
member0_.team_id as team_id4_0_,
member0_.user_name as user_nam3_0_
from
member member0_
where
member0_.age=?
order by
member0_.user_name desc limit ?
select
count(member0_.member_id) as col_0_0_
from
member member0_
where
member0_.age=?
5. Slice
- Slice : 요청할 때 limit + 1개 더, 그래서 전체 count가져오지 않음.
- 반환 타입을 Page -> Slice로 바꿨더니 getTotalElements를 사용할 수 없다!!
- Repository
Slice<Member> findByAge(int age, Pageable pageable);
- 참고로 Page는 Slice를 상속받는다. 반환 타입 설정에 주의!!
- Test
@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(Sort.Direction.DESC, "userName"));
//when
//Page<Member> page = memberRepository.findByAge(age, pageRequest);
Slice<Member> page = memberRepository.findByAge(age, pageRequest);
//then
List<Member> content = page.getContent(); //data 꺼내기
//long totalCount = page.getTotalElements(); //totalCount
Assertions.assertThat(content.size()).isEqualTo(3);
//Assertions.assertThat(page.getTotalElements()).isEqualTo(5);
Assertions.assertThat(page.getNumber()).isEqualTo(0); //page 번호
//Assertions.assertThat(page.getTotalPages()).isEqualTo(2);
Assertions.assertThat(page.isFirst()).isTrue();
Assertions.assertThat(page.hasNext()).isTrue();
}
- slice는 limit + 1이라고 하는데 진짜로 그럴까?
- 위의 코드를 보면 limit는 3으로 지정했는데... 로그를 보면 4로 들어가있다 ㅋㅋㅋㅋ
select
member0_.member_id as member_i1_0_,
member0_.age as age2_0_,
member0_.team_id as team_id4_0_,
member0_.user_name as user_nam3_0_
from
member member0_
where
member0_.age=?
order by
member0_.user_name desc limit ?
<p6spy>
select
member0_.member_id as member_i1_0_,
member0_.age as age2_0_,
member0_.team_id as team_id4_0_,
member0_.user_name as user_nam3_0_
from
member member0_
where
member0_.age=10
order by
member0_.user_name desc limit 4;
- Slice를 사용하면 미리 하나 더 불러와서 "더보기"누르면 하나 보여주도로 만듦 ㅋㅋㅋ
6. Query와 CountQuery 분리
- totalCount는 결국 DB에 있는 전체 DATA수를 계산해야함. 즉 성능이 안나옴.
- data가 많아질수록 답없네..
- Query가 복잡하면 ContQuery도 복잡해짐...
- member와 team을 left outer join한다고 해보자.
@Query(value = "select m from Member m left join m.team t")
Page<Member> findByAge(int age, Pageable pageable);
- 그런데 Count쿼리는 join이 필요없다? 왜? 조인 하던 말던 member의 수에는 관계 없다. ㅋㅋㅋ
- 근데 Count쿼리도 join이 날라간다.
select
member0_.member_id as member_i1_0_,
member0_.age as age2_0_,
member0_.team_id as team_id4_0_,
member0_.user_name as user_nam3_0_
from
member member0_
left outer join
team team1_
on member0_.team_id=team1_.team_id
order by
member0_.user_name desc limit ?
:
select
count(member0_.member_id) as col_0_0_
from
member member0_
left outer join
team team1_
on member0_.team_id=team1_.team_id
- 최적화다 필요하다. CountQuery를 따로 분리 할 피요가 있다.
- CountQuery를 별도로 작성하며 단순하게 member의 수만 확인하도록 하자.
@Query(value = "select m from Member m left join m.team t",countQuery = "select count(m.userName) from Member m")
Page<Member> findByAge(int age, Pageable pageable);
- 크 CountQuery가 심플해졌다.
select
count(member0_.user_name) as col_0_0_
from
member member0_
7. 참고
- Page는 1부터 시작이 아니라 0부터 시작이다.
- Paging에서 추가로 제공하는 정보 필요없이 단순이 data끊어서 가져오려면 List<>로 받아도 가능.
- Page<Entity>는 API에 노출하면 안돼!!!
- DTO로 변환해야함!
- page.map을 활용하면 됨.
- 현재 content를 주어진 function에 의해 다른 content로 바꾼 Page를 반환함.
Page<MemberDto> toMap = page.map(member -> new MemberDto(member.getId(), member.getUserName(), null));
8. GitHub : 210829 SpringDataJpa, Paging
'인프런 > [인프런] Spring Data JPA' 카테고리의 다른 글
@EntityGraph (0) | 2021.08.29 |
---|---|
Query Method, 벌크성 수정 쿼리 (0) | 2021.08.29 |
JPA의 페이징과 정렬 (0) | 2021.08.29 |
@Query, 반환 타입 (0) | 2021.08.29 |
@Query, 파라미터 바인딩 (0) | 2021.08.29 |
Comments