web-dev-qa-db-fra.com

Impossible de lire JSON: impossible de désérialiser l'instance de hello.Country [] à partir du jeton START_OBJECT

J'ai l'URL de repos qui me donne tous les pays - http://api.geonames.org/countryInfoJSON?username=volodiaL .

J'utilise RestTemplate à partir du printemps 3 pour analyser json renvoyé dans les objets Java:

RestTemplate restTemplate = new RestTemplate();
Country[] countries = restTemplate.getForObject("http://api.geonames.org/countryInfoJSON?username=volodiaL",Country[].class);

Lorsque je lance ce code, je reçois une exception:

Caused by: com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of hello.Country[] out of START_OBJECT token
 at [Source: Sun.net.www.protocol.http.HttpURLConnection$HttpInputStream@1846149; line: 1, column: 1]
    at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.Java:164)
    at com.fasterxml.jackson.databind.DeserializationContext.mappingException(DeserializationContext.Java:691)
    at com.fasterxml.jackson.databind.DeserializationContext.mappingException(DeserializationContext.Java:685)
    at com.fasterxml.jackson.databind.deser.std.ObjectArrayDeserializer.handleNonArray(ObjectArrayDeserializer.Java:222)
    at com.fasterxml.jackson.databind.deser.std.ObjectArrayDeserializer.deserialize(ObjectArrayDeserializer.Java:133)
    at com.fasterxml.jackson.databind.deser.std.ObjectArrayDeserializer.deserialize(ObjectArrayDeserializer.Java:18)
    at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.Java:2993)
    at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.Java:2158)
    at org.springframework.http.converter.json.MappingJackson2HttpMessageConverter.readJavaType(MappingJackson2HttpMessageConverter.Java:225)
    ... 7 more

Enfin ma classe de pays:

import com.fasterxml.jackson.annotation.JsonIgnoreProperties;

@JsonIgnoreProperties(ignoreUnknown = true)
public class Country {
    private String countryName;
    private long geonameId;

    public String getCountryName() {
        return countryName;
    }

    public long getGeonameId() {
        return geonameId;
    }

    @Override
    public String toString() {
        return countryName;
    }
}

Le problème est que json retourné contient l’élément racine "geonames" qui contient un tableau d’éléments de pays, comme ceci:

{
"geonames": [
    {
        "continent": "EU",
        "capital": "Andorra la Vella",
        "languages": "ca",
        "geonameId": 3041565,
        "south": 42.42849259876837,
        "isoAlpha3": "AND",
        "north": 42.65604389629997,
        "fipsCode": "AN",
        "population": "84000",
        "east": 1.7865427778319827,
        "isoNumeric": "020",
        "areaInSqKm": "468.0",
        "countryCode": "AD",
        "west": 1.4071867141112762,
        "countryName": "Andorra",
        "continentName": "Europe",
        "currencyCode": "EUR"
    }
]
}

Comment dire à RestTemplate de convertir chaque élément du tableau en un objet Country?

36
Volodymyr Levytskyi

Vous devez faire ce qui suit:

public class CountryInfoResponse {

   @JsonProperty("geonames")
   private List<Country> countries; 

   //getter - setter
}

RestTemplate restTemplate = new RestTemplate();
List<Country> countries = restTemplate.getForObject("http://api.geonames.org/countryInfoJSON?username=volodiaL",CountryInfoResponse.class).getCountries();

Ce serait bien si vous pouviez utiliser une sorte d'annotation pour vous permettre de sauter des niveaux, mais ce n'est pas encore possible (voir this et this )

41
geoand

Une autre solution:

public class CountryInfoResponse {
  private List<Object> geonames;
}

Utilisation d’un nom générique Object - La liste a résolu mon problème, car il existait d’autres types de données tels que Boolean.

9
vonox7

Dans mon cas, jQuery valait <input type="text"> et je l’ai fait comme ceci:

var newUserInfo = { "lastName": inputLastName[0].value, "userName": inputUsername[0].value,
 "firstName": inputFirstName[0] , "email": inputEmail[0].value}

Et je recevais constamment cette exception

com.fasterxml.jackson.databind.JsonMappingException: impossible de désérialiser l'instance de Java.lang.String en dehors du jeton START_OBJECT à l'adresse [Source: Java.io.PushbackInputStream@39cb6c98; ligne: 1, colonne: 54] (via la chaîne de référence: com.springboot.domain.User ["firstName"]).

Et je me suis cogné la tête pendant environ une heure avant de me rendre compte que j'avais oublié d'écrire .value après ceci"firstName": inputFirstName[0].

La solution correcte était donc:

var newUserInfo = { "lastName": inputLastName[0].value, "userName": inputUsername[0].value,
 "firstName": inputFirstName[0].value , "email": inputEmail[0].value}

Je suis venu ici parce que j'avais ce problème et j'espère sauver à quelqu'un d'autre des heures de misère.

À votre santé :)

3
Filip Savic

Si vous voulez éviter d'utiliser un Class et List<Object> genomes supplémentaires, vous pouvez simplement utiliser un Map.

La structure de données se traduit par Map<String, List<Country>>

String resourceEndpoint = "http://api.geonames.org/countryInfoJSON?username=volodiaL";

Map<String, List<Country>> geonames = restTemplate.getForObject(resourceEndpoint, Map.class);

List<Country> countries = geonames.get("geonames");
0
clD