RestTemplate:无法从START_OBJECT标记中反序列化OBJECT的实例

时间:2015-01-20 21:07:35

标签: java json spring jackson resttemplate

我从春天开始尝试RestTemplate的东西。我试图读取这个JSON数据:JSON Data。数据是一个键值对,其中键是" geonames"并且值是" geoname"对象。

我有一个Geoname类来处理输入。这个类里面还有getter和setter。然后我有一个app类,它只运行一个main方法来调用RestTemplate对象:

@JsonIgnoreProperties(ignoreUnknown = true)
public class Geoname {
    private String name;
    private long lat;
    private long lng;
    private String countrycode;
}

App.java

public class App 
{
    public static void main( String[] args )
    {
        String jsonUrl = "http://api.geonames.org/citiesJSON?north=44.1&south=-9.9&east=-22.4&west=55.2&lang=de&username=demo";

        RestTemplate template = new RestTemplate();
        ResponseEntity<Geoname[]> entity = template.getForEntity(jsonUrl, Geoname[].class);
        List<Geoname> data = Arrays.asList(entity.getBody());

        System.out.print("Success!");
    }
}

这是我的错误输出:

Exception in thread "main" org.springframework.http.converter.HttpMessageNotReadableException: Could not read JSON: Can not deserialize instance of com.declan.Geoname[] out of START_OBJECT token
 at [Source: sun.net.www.protocol.http.HttpURLConnection$HttpInputStream@54fc3ac5; line: 1, column: 1]; nested exception is com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of com.declan.Geoname[] out of START_OBJECT token
 at [Source: sun.net.www.protocol.http.HttpURLConnection$HttpInputStream@54fc3ac5; line: 1, column: 1]
    at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.readJavaType(AbstractJackson2HttpMessageConverter.java:208)
    at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.read(AbstractJackson2HttpMessageConverter.java:200)
    at org.springframework.web.client.HttpMessageConverterExtractor.extractData(HttpMessageConverterExtractor.java:96)
    at org.springframework.web.client.RestTemplate$ResponseEntityResponseExtractor.extractData(RestTemplate.java:812)
    at org.springframework.web.client.RestTemplate$ResponseEntityResponseExtractor.extractData(RestTemplate.java:796)
    at org.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:576)
    at org.springframework.web.client.RestTemplate.execute(RestTemplate.java:529)
    at org.springframework.web.client.RestTemplate.getForEntity(RestTemplate.java:261)
    at com.declan.App.main(App.java:20)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)
Caused by: com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of com.declan.Geoname[] out of START_OBJECT token
 at [Source: sun.net.www.protocol.http.HttpURLConnection$HttpInputStream@54fc3ac5; line: 1, column: 1]
    at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:148)
    at com.fasterxml.jackson.databind.DeserializationContext.mappingException(DeserializationContext.java:835)
    at com.fasterxml.jackson.databind.DeserializationContext.mappingException(DeserializationContext.java:831)
    at com.fasterxml.jackson.databind.deser.std.ObjectArrayDeserializer.handleNonArray(ObjectArrayDeserializer.java:232)
    at com.fasterxml.jackson.databind.deser.std.ObjectArrayDeserializer.deserialize(ObjectArrayDeserializer.java:139)
    at com.fasterxml.jackson.databind.deser.std.ObjectArrayDeserializer.deserialize(ObjectArrayDeserializer.java:17)
    at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:3560)
    at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:2660)
    at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.readJavaType(AbstractJackson2HttpMessageConverter.java:205)
    ... 13 more

我已尝试过此设置,我尝试过使用类似于弹簧文档的getForObject,我已尝试在此处搜索stackoverflow并使用映射到列表的常见答案。我甚至尝试过创建一个Geonames类,其中只包含一个数组或Geoname对象但是它们都没有工作......始终存在相同的错误。也许,我没有正确地阅读Json,但如果有人能借给我一双眼睛,我会感激不尽。 :D

干杯!

修改

好的,我现在有了这个新类,它现在发出了对JSON数据的GET请求。但是,在调试之后,ResponseEntity主体将数组设置为null。我是否需要手动在Geonames中实例化数组?

@JsonIgnoreProperties(ignoreUnknown = true)
public class Geonames {
    @JsonProperty("geonames")
    Geoname[] geonames;

    public void setGeonames(Geonames[] geonames) {
        this.geonames = geonames;
    }

    public void getGeonames() {
        return geonames;
    }
}

1 个答案:

答案 0 :(得分:1)

解决。事实证明,null是因为JSON链接具有每小时最大命中数限制。通过在网站上创建我自己的帐户,api给了我自己的限制。因此响应主体随后填充了数据。