Ошибка при использовании Джексона и Джсона
Я пытаюсь проанализировать некоторые JSON (полный пример JSON можно увидеть в этом Gist). Я показываю общую структуру JSON ниже:
[
{
"title": "Principles of Compiler Design",
"authors": [
"Aho",
"Ullman"
],
"publisher": "Addison Wesley",
"year": 1977
},
{
"title": "Compilers: Principles Techniques and Tools",
"authors": [
"Aho",
"Sethi",
"Ullman"
],
"publisher": "Addison Wesley",
"year": 1985
}
]
Я пытаюсь проанализировать JSON с библиотеками Джексона, но при тестировании получаю следующую ошибку:
Exception in thread "main" com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of java.lang.String out of START_ARRAY token
at [Source: library.json; line: 2, column: 49] (through reference chain: com.acme.datatypes.User["authors"])
at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:163)
at com.fasterxml.jackson.databind.DeserializationContext.mappingException(DeserializationContext.java:588)
at com.fasterxml.jackson.databind.deser.std.JdkDeserializers$StringDeserializer.deserialize(JdkDeserializers.java:90)
at com.fasterxml.jackson.databind.deser.std.JdkDeserializers$StringDeserializer.deserialize(JdkDeserializers.java:59)
at com.fasterxml.jackson.databind.deser.SettableBeanProperty.deserialize(SettableBeanProperty.java:336)
at com.fasterxml.jackson.databind.deser.impl.MethodProperty.deserializeAndSet(MethodProperty.java:89)
at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromObject(BeanDeserializer.java:290)
at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:112)
at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:2563)
at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:1759)
at com.acme.datatypes.UserTest.main(UserTest.java:20)
Вот мой код:
Класс пользовательского теста:
public class UserTest {
public static void main(String[] args) throws JsonParseException,
JsonMappingException, IOException {
File jsonFile = new File("library.json");
User user = null;
ObjectMapper mapper = new ObjectMapper();
user = mapper.readValue(jsonFile, User.class);
System.out.println(user.getTitle());
user = mapper.readValue(jsonFile, User.class);
System.out.println(user.getAuthors());
user = mapper.readValue(jsonFile, User.class);
System.out.println(user.getPublisher());
user = mapper.readValue(jsonFile, User.class);
System.out.println(user.getYear());
}
}
Класс пользователя:
public class User {
private String authors;
private String publisher;
private String title;
private Number year;
public String getAuthors() {
return this.authors;
}
public void setAuthors(String authors) {
this.authors = authors;
}
public String getPublisher() {
return this.publisher;
}
public void setPublisher(String publisher) {
this.publisher = publisher;
}
public String getTitle() {
return this.title;
}
public void setTitle(String title) {
this.title = title;
}
public Number getYear() {
return this.year;
}
public void setYear(Number year) {
this.year = year;
}
}
Кто-нибудь знает, в чем может быть проблема? Благодарю.
2 ответа
Две быстрые вещи:
Ваш пользовательский класс определяет
authors
свойство в виде строки. Но в JSON это массив, поэтому вам нужно использовать коллекции или тип массива в вашем Java-объекте. Что-то вроде:private List<String> authors
Вы неоднократно анализируете файл JSON в своем тестовом классе. Вам нужно только проанализировать его один раз, и вам нужно использовать токен супертипа, поскольку в JSON есть список элементов (а не только один). Вы также используете неправильный тип для десериализации (User.class). Вместо всех этих строк:
user = mapper.readValue(jsonFile, User.class);
System.out.println(user.getTitle());
user = mapper.readValue(jsonFile, User.class); // <-- unnecessary parsing System.out.println(user.getAuthors());
user = mapper.readValue(jsonFile, User.class); // <-- unnecessary parsing System.out.println(user.getPublisher());
user = mapper.readValue(jsonFile, User.class); // <-- unnecessary parsing System.out.println(user.getYear());
Просто используйте:
List<User> userList =
mapper.readValue(jsonFile, new TypeReference<List<User>>() {});
Как только вы получите список пользователей в вашем тестовом классе, вы можете выполнять их итерацию, используя расширенный цикл for.
for(User user : userList) {
System.out.println(user.getTitle());
}
Поскольку вы работаете с массивом, вам необходимо преобразовать его в массив или список
Как массив
MyClass[] myObjects = mapper.readValue(json, MyClass[].class);
Как список
List<MyClass> myObjects = mapper.readValue(jsonInput, new TypeReference<List<MyClass>>(){});
пользователь
public class User {
private List<String> authors;
private String publisher;
private String title;
private Number year;
public List<String> getAuthors() {
return this.authors;
}
public void setAuthors(List<String> authors) {
this.authors = authors;
}
public String getPublisher() {
return this.publisher;
}
public void setPublisher(String publisher) {
this.publisher = publisher;
}
public String getTitle() {
return this.title;
}
public void setTitle(String title) {
this.title = title;
}
public Number getYear() {
return this.year;
}
public void setYear(Number year) {
this.year = year;
}
}
Использование:
List<User> l = mapper.readValue(new File(""),new TypeReference<List<User>>() {});