我们的REST API在Pages中返回结果。以下是一个Controller的示例
@RequestMapping(value = "/search", method = RequestMethod.GET, produces = MediaType.APPLICATION_JSON_VALUE + ";charset=UTF-8")
@ResponseStatus(HttpStatus.OK)
public Page<MyObject> findAll(Pageable pageable) {
...
}
使用RestTemplate有一种简单的方法来使用该API吗?
如果我们这样做
ParameterizedTypeReference<Page<MyObject>> responseType = new ParameterizedTypeReference<Page<MyObject>>() { };
ResponseEntity<Page<MyObject>> result = restTemplate.exchange(url, HttpMethod.GET, null/*httpEntity*/, responseType);
List<MyObject> searchResult = result.getBody().getContent();
它抛出异常
org.springframework.http.converter.HttpMessageNotReadableException: Could not read document: Can not construct instance of org.springframework.data.domain.Page,
problem: abstract types either need to be mapped to concrete types, have custom deserializer, or be instantiated with additional type information at [Source: java.io.PushbackInputStream@3be1e1f2; line: 1, column: 1]; nested exception is com.fasterxml.jackson.databind.JsonMappingException: Can not construct instance of org.springframework.data.domain.Page, problem: abstract types either need to be mapped to concrete types, have custom deserializer, or be instantiated with additional type information
提前谢谢
答案 0 :(得分:17)
从Spring Boot 1.x迁移到2.0时, 将读取Rest API响应的代码更改为
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.JsonNode;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import java.util.ArrayList;
import java.util.List;
public class RestPageImpl<T> extends PageImpl<T>{
@JsonCreator(mode = JsonCreator.Mode.PROPERTIES)
public RestPageImpl(@JsonProperty("content") List<T> content,
@JsonProperty("number") int number,
@JsonProperty("size") int size,
@JsonProperty("totalElements") Long totalElements,
@JsonProperty("pageable") JsonNode pageable,
@JsonProperty("last") boolean last,
@JsonProperty("totalPages") int totalPages,
@JsonProperty("sort") JsonNode sort,
@JsonProperty("first") boolean first,
@JsonProperty("numberOfElements") int numberOfElements) {
super(content, PageRequest.of(number, size), totalElements);
}
public RestPageImpl(List<T> content, Pageable pageable, long total) {
super(content, pageable, total);
}
public RestPageImpl(List<T> content) {
super(content);
}
public RestPageImpl() {
super(new ArrayList<>());
}
}
答案 1 :(得分:13)
将读取Rest API响应的代码更改为;
ParameterizedTypeReference<RestResponsePage<MyObject>> responseType = new ParameterizedTypeReference<RestResponsePage<MyObject>>() { };
ResponseEntity<RestResponsePage<MyObject>> result = restTemplate.exchange(url, HttpMethod.GET, null/*httpEntity*/, responseType);
List<MyObject> searchResult = result.getBody().getContent();
这是我为RestResponsePage
创建的类package com.basf.gb.cube.seq.vaadinui.util;
import java.util.ArrayList;
import java.util.List;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.Pageable;
public class RestResponsePage<T> extends PageImpl<T>{
private static final long serialVersionUID = 3248189030448292002L;
public RestResponsePage(List<T> content, Pageable pageable, long total) {
super(content, pageable, total);
// TODO Auto-generated constructor stub
}
public RestResponsePage(List<T> content) {
super(content);
// TODO Auto-generated constructor stub
}
/* PageImpl does not have an empty constructor and this was causing an issue for RestTemplate to cast the Rest API response
* back to Page.
*/
public RestResponsePage() {
super(new ArrayList<T>());
}
}
答案 2 :(得分:8)
扩展上面,但不需要实现每个属性。
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import java.util.ArrayList;
import java.util.List;
public class RestPageImpl<T> extends PageImpl<T>{
@JsonCreator(mode = JsonCreator.Mode.PROPERTIES)
public RestPageImpl(@JsonProperty("content") List<T> content,
@JsonProperty("number") int page,
@JsonProperty("size") int size,
@JsonProperty("totalElements") long total) {
super(content, new PageRequest(page, size), total);
}
public RestPageImpl(List<T> content, Pageable pageable, long total) {
super(content, pageable, total);
}
public RestPageImpl(List<T> content) {
super(content);
}
public RestPageImpl() {
super(new ArrayList());
}
}
答案 3 :(得分:3)
无需实施Page
。您只需在PagedResources<T>
中使用ParameterizedTypeReference
作为类型。
因此,如果您的服务返回类似于(为简洁起见而删除对象)的响应:
{
"_embedded": {
"events": [
{...},
{...},
{...},
{...},
{...}
]
},
"_links": {
"first": {...},
"self": {...},
"next": {...},
"last": {...}
},
"page": {
"size": 5,
"totalElements": 30,
"totalPages": 6,
"number": 0
}
}
您关注的对象属于Event
类型,那么您应该执行以下请求:
ResponseEntity<PagedResources<Event>> eventsResponse = restTemplate.exchange(uriBuilder.build(true).toUri(),
HttpMethod.GET, null, new ParameterizedTypeReference<PagedResources<Event>>() {});
如果您掌握了以下资源:
PagedResources<Event> eventsResources = eventsResponse.getBody();
您将能够访问页面元数据(您在"page"
部分中获得的内容),链接("_links"
部分)和内容:
Collection<Event> eventsCollection = eventsResources.getContent();
答案 4 :(得分:0)
发布的解决方案对我没有用,因为总元素设置不正确。我使用委托模式实现了页面,这对我有用。这是工作代码:
import org.springframework.core.convert.converter.Converter;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
public class RestPage<T> implements Page<T> {
private PageImpl<T> pageDelegate = new PageImpl<>(new ArrayList<>(0));
public List<T> getContent() {
return pageDelegate.getContent();
}
public int getNumber() {
return pageDelegate.getNumber();
}
public int getNumberOfElements() {
return pageDelegate.getNumberOfElements();
}
public int getSize() {
return pageDelegate.getSize();
}
public Sort getSort() {
return pageDelegate.getSort();
}
public long getTotalElements() {
return pageDelegate.getTotalElements();
}
public int getTotalPages() {
return pageDelegate.getTotalPages();
}
public boolean hasContent() {
return pageDelegate.hasContent();
}
public boolean hasNext() {
return pageDelegate.hasNext();
}
public boolean hasPrevious() {
return pageDelegate.hasPrevious();
}
public boolean isFirst() {
return pageDelegate.isFirst();
}
public boolean isLast() {
return pageDelegate.isLast();
}
public Iterator<T> iterator() {
return pageDelegate.iterator();
}
public <S> Page<S> map(Converter<? super T, ? extends S> converter) {
return pageDelegate.map(converter);
}
public Pageable nextPageable() {
return pageDelegate.nextPageable();
}
public Pageable previousPageable() {
return pageDelegate.previousPageable();
}
public void setContent(List<T> content) {
pageDelegate = new PageImpl<>(content, null, getTotalElements());
}
public void setTotalElements(int totalElements) {
pageDelegate = new PageImpl<>(getContent(), null, totalElements);
}
public String toString() {
return pageDelegate.toString();
}
}
答案 5 :(得分:0)
科特林解决方案
import com.fasterxml.jackson.annotation.JsonCreator
import com.fasterxml.jackson.annotation.JsonProperty
import com.fasterxml.jackson.databind.JsonNode
import org.springframework.data.domain.PageImpl
import org.springframework.data.domain.PageRequest
import org.springframework.data.domain.Pageable
import java.util.ArrayList
class RestResponsePage<T> : PageImpl<T> {
@JsonCreator(mode = JsonCreator.Mode.PROPERTIES)
constructor(@JsonProperty("content") content: List<T>,
@JsonProperty("number") number: Int,
@JsonProperty("size") size: Int,
@JsonProperty("totalElements") totalElements: Long?,
@JsonProperty("pageable") pageable: JsonNode,
@JsonProperty("last") last: Boolean,
@JsonProperty("totalPages") totalPages: Int,
@JsonProperty("sort") sort: JsonNode,
@JsonProperty("first") first: Boolean,
@JsonProperty("numberOfElements") numberOfElements: Int) : super(content, PageRequest.of(number, size), totalElements!!) {
}
constructor(content: List<T>, pageable: Pageable, total: Long) : super(content, pageable, total) {}
constructor(content: List<T>) : super(content) {}
constructor() : super(ArrayList<T>()) {}
}
并请求
var response: ResponseEntity<*> = restTemplate.getForEntity<RestResponsePage<SomeObject>>(url)
Java解决方案
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.JsonNode;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import java.util.ArrayList;
import java.util.List;
public class RestResponsePage<T> extends PageImpl<T> {
@JsonCreator(mode = JsonCreator.Mode.PROPERTIES)
public RestResponsePage(@JsonProperty("content") List<T> content,
@JsonProperty("number") int number,
@JsonProperty("size") int size,
@JsonProperty("totalElements") Long totalElements,
@JsonProperty("pageable") JsonNode pageable,
@JsonProperty("last") boolean last,
@JsonProperty("totalPages") int totalPages,
@JsonProperty("sort") JsonNode sort,
@JsonProperty("first") boolean first,
@JsonProperty("numberOfElements") int numberOfElements) {
super(content, PageRequest.of(number, size), totalElements);
}
public RestResponsePage(List<T> content, Pageable pageable, long total) {
super(content, pageable, total);
}
public RestResponsePage(List<T> content) {
super(content);
}
public RestResponsePage() {
super(new ArrayList<>());
}
}
答案 6 :(得分:0)
我必须做一个很小的更改,以便它可以忽略似乎是最近引入的empty的未知属性。
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.JsonNode;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import java.util.ArrayList;
import java.util.List;
@JsonIgnoreProperties(ignoreUnknown = true)
public class RestResponsePage<T> extends PageImpl<T> {
@JsonCreator(mode = JsonCreator.Mode.PROPERTIES)
public RestResponsePage(@JsonProperty("content") List<T> content,
@JsonProperty("number") int number,
@JsonProperty("size") int size,
@JsonProperty("totalElements") Long totalElements,
@JsonProperty("pageable") JsonNode pageable,
@JsonProperty("last") boolean last,
@JsonProperty("totalPages") int totalPages,
@JsonProperty("sort") JsonNode sort,
@JsonProperty("first") boolean first,
@JsonProperty("numberOfElements") int numberOfElements) {
super(content, PageRequest.of(number, size), totalElements);
}
public RestResponsePage(List<T> content, Pageable pageable, long total) {
super(content, pageable, total);
}
public RestResponsePage(List<T> content) {
super(content);
}
public RestResponsePage() {
super(new ArrayList<>());
}
}
答案 7 :(得分:0)
如果您使用 FeignClient,有一个简单的解决方案,即创建一个配置类。
作者: https://github.com/spring-cloud/spring-cloud-openfeign/issues/205
配置类:
public class FeignDecodeConfiguration {
@Bean
public Module pageJacksonModule () {
return new PageJacksonModule ();
}
}