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
?
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 )
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.
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é :)
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");