Spring Rest Template를 이용한 페이지 응답 소비 방법
스프링 데이터(mongoDb)를 사용하고 있으며 저장소를 가지고 있습니다.
public interface StoriesRepository extends PagingAndSortingRepository<Story, String> {}
그리고 컨트롤러가 있습니다.
@RequestMapping(method = RequestMethod.GET)
public ResponseEntity<Page<StoryResponse>> getStories(Pageable pageable) {
Page<StoryResponse> stories = storiesRepository.findAll(pageable).map(StoryResponseMapper::toStoryResponse);
return ResponseEntity.ok(stories);
}
모든 것이 정상적으로 작동하지만 RestTemplate getForEntity 메서드를 사용하여 엔드포인트를 사용할 수 없습니다.
def entity = restTemplate.getForEntity(getLocalhost("/story"), new TypeReference<Page<StoryResponse>>(){}.class)
엔티티 페이지를 성공적으로 역직렬화하려면 어떤 클래스를 제공해야 합니까?
new TypeReference<Page<StoryResponse>>() {}
이 문장의 문제는 Jackson이 추상 형식을 인스턴스화할 수 없다는 것입니다.당신은 잭슨에게 어떻게 인스턴스화하는지에 대한 정보를 주어야 합니다.Page
구체적인 활자로하지만 구체적인 유형은PageImpl
기본 생성자 또는 모든 생성자가 없습니다.@JsonCreator
s. 따라서 다음 코드도 사용할 수 없습니다.
new TypeReference<PageImpl<StoryResponse>>() {}
필요한 정보를 추가할 수 없기 때문에Page
클래스, 사용자 정의 구현을 생성하는 것이 좋습니다.Page
이 답변에서와 같이 기본 no-arg 생성자가 있는 인터페이스.그런 다음 유형 참조에서 다음과 같은 사용자 정의 구현을 사용합니다.
new TypeReference<CustomPageImpl<StoryResponse>>() {}
다음은 연결된 질문에서 복사한 사용자 지정 구현입니다.
public class CustomPageImpl<T> extends PageImpl<T> {
private static final long serialVersionUID = 1L;
private int number;
private int size;
private int totalPages;
private int numberOfElements;
private long totalElements;
private boolean previousPage;
private boolean firstPage;
private boolean nextPage;
private boolean lastPage;
private List<T> content;
private Sort sort;
public CustomPageImpl() {
super(new ArrayList<>());
}
@Override
public int getNumber() {
return number;
}
public void setNumber(int number) {
this.number = number;
}
@Override
public int getSize() {
return size;
}
public void setSize(int size) {
this.size = size;
}
@Override
public int getTotalPages() {
return totalPages;
}
public void setTotalPages(int totalPages) {
this.totalPages = totalPages;
}
@Override
public int getNumberOfElements() {
return numberOfElements;
}
public void setNumberOfElements(int numberOfElements) {
this.numberOfElements = numberOfElements;
}
@Override
public long getTotalElements() {
return totalElements;
}
public void setTotalElements(long totalElements) {
this.totalElements = totalElements;
}
public boolean isPreviousPage() {
return previousPage;
}
public void setPreviousPage(boolean previousPage) {
this.previousPage = previousPage;
}
public boolean isFirstPage() {
return firstPage;
}
public void setFirstPage(boolean firstPage) {
this.firstPage = firstPage;
}
public boolean isNextPage() {
return nextPage;
}
public void setNextPage(boolean nextPage) {
this.nextPage = nextPage;
}
public boolean isLastPage() {
return lastPage;
}
public void setLastPage(boolean lastPage) {
this.lastPage = lastPage;
}
@Override
public List<T> getContent() {
return content;
}
public void setContent(List<T> content) {
this.content = content;
}
@Override
public Sort getSort() {
return sort;
}
public void setSort(Sort sort) {
this.sort = sort;
}
public Page<T> pageImpl() {
return new PageImpl<>(getContent(), new PageRequest(getNumber(),
getSize(), getSort()), getTotalElements());
}
}
이 실이 조금 오래된 것은 알지만, 누군가가 이것을 통해 이익을 얻을 수 있기를 바랍니다.
@알리 데하니의 대답은 좋지만, 그것이 무엇을 다시 구현한다는 것을 제외하고는.PageImpl<T>
이미 완료했습니다.저는 이것이 다소 불필요하다고 생각했습니다.확장되는 클래스를 만들어 더 나은 솔루션을 찾았습니다.PageImpl<T>
지정합니다.@JsonCreator
생성자:
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.company.model.HelperModel;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.PageRequest;
import java.util.List;
public class HelperPage extends PageImpl<HelperModel> {
@JsonCreator
// Note: I don't need a sort, so I'm not including one here.
// It shouldn't be too hard to add it in tho.
public HelperPage(@JsonProperty("content") List<HelperModel> content,
@JsonProperty("number") int number,
@JsonProperty("size") int size,
@JsonProperty("totalElements") Long totalElements) {
super(content, new PageRequest(number, size), totalElements);
}
}
그러면:
HelperPage page = restTemplate.getForObject(url, HelperPage.class);
이것은 다음을 만드는 것과 같습니다.CustomPageImpl<T>
클래스이지만 이미 있는 모든 코드를 활용할 수 있습니다.PageImpl<T>
.
"경로 탐지기"가 언급했듯이, 당신은 사용할 수 있습니다.exchange
의 방법RestTemplate
하지만 지나가는 대신에ParameterizedTypeReference<Page<StoryResponse>>()
당신은 통과해야 합니다ParameterizedTypeReference<PagedResources<StoryResponse>>()
응답을 받으면 내용을 검색할 수 있습니다.Collection<StoryResponse>
.
코드는 다음과 같아야 합니다.
ResponseEntity<PagedResources<StoryResponse>> response = restTemplate.exchange(getLocalhost("/story"),
HttpMethod.GET, null, new ParameterizedTypeReference<PagedResources<StoryResponse>>() {});
PagedResources<StoryResponse> storiesResources = response.getBody();
Collection<StoryResponse> stories = storiesResources.getContent();
내용은 차치하고storiesResources
페이지 메타데이터 및 링크도 포함합니다.
자세한 설명은 https://stackoverflow.com/a/46847429/8805916 에서 확인할 수 있습니다.
spring-cloud-open feature를 사용하면 PageJackson Module을 사용할 수 있습니다.객체 매퍼에 PageJacksonModule을 등록하기만 하면 됩니다.
final ObjectMapper mapper = new ObjectMapper()
mapper.registerModule(new PageJacksonModule());
만약 당신이 이 스레드를 보고, 그리고 당신이 이것을 시도한다면 https://stackoverflow.com/a/44895867/8268335 에 답하세요.
두 번째 문제를 해결할 수 있습니다.
Can not construct instance of org.springframework.data.domain.Pageable
그러면 여기서 완벽한 솔루션을 찾을 수 있습니다. https://stackoverflow.com/a/42002709/8268335
.RestPageImpl
위의 답변에서 문제를 해결했습니다.
restTemplate의 교환 방법을 사용하여 본문을 얻을 수 있습니다.
다음 답변을 확인하십시오. https://stackoverflow.com/a/31947188/3800576 .이것은 당신에게 도움이 될 것입니다.
언급URL : https://stackoverflow.com/questions/34099559/how-to-consume-pageentity-response-using-spring-resttemplate
'source' 카테고리의 다른 글
clock_gettime()의 다양한 클럭 이해 (0) | 2023.07.26 |
---|---|
JavaScript/JQuery를 사용하여 간단한 맵을 만드는 방법 (0) | 2023.07.26 |
Maria Db - 사용자 ID가 = 변수인 모든 메시지 목록에서 최근 메시지를 가져옵니다. (0) | 2023.07.26 |
Powershell 7.3.0 탭 완료가 작동하지 않음 (0) | 2023.07.26 |
Angular 2의 링크에 접두사 "unsafe"를 추가하지 않으려면 어떻게 해야 합니까? (0) | 2023.07.26 |