如何使用Genson解析嵌套的json?

时间:2019-05-13 14:04:16

标签: json parsing nested genson

我从REST API获取结果,它是代表用户的json对象列表,每个用户内部都有嵌套的json对象。我的问题是嵌套的属性名称与我在代码中获得的bean属性不对应。但是它们太不合适了,以至于我真的不想保留API嵌套的属性名称...

我正在将Genson 1.5与Java 8和lombok用于我的豆子。 我尝试使用简单的反序列化器,然后使用Converter,但没有成功。

以下是我从API收到的示例:

[
   {
      "FirstName": "Jack",
      "LastName": "Sparrow",
      "Adress": {
                   "String1": "Toulon",
                   "String2": "France",
                   "String3": "83",
                }
   },
   {
      "FirstName": "I am",
      "LastName": "Groot",
      "Adress": {
                   "String1": "Marseille",
                   "String2": "France",
                   "String3": "13",
                }
   },
]

这是我想要的一种新格式的豆子:

@Data
public class User {
   private String firstName;
   private String lastName;
   private String country; //this represents String2 from the API
}

我已经尝试过这些解决方案:

  • 仅使用推荐的反序列化(之所以出错,是因为与我的bean相比,api不能获得相同的属性):
private Genson genson = new Genson();

public List<User> getUserList() {
   Response response = api.target(url)...get();
   List<User> users = genson.deserialize(response.readEntity(String.class), new GenericType<List<User>>(){});
   return users;
}
  • 使用转换器获取与我的bean相同的参数
private Genson genson = new GensonBuilder().withConverters(new UserConverter()).create();

public List<User> getUserList() {
   Response response = api.target(url)...get();
   List<User> users = this.genson.deserialize(response.readEntity(String.class), new genericType<List<User>>(){});
   return users;
}
public class UserConverter implements Converter<User> {

    public User deserialize(ObjectReader reader, Context ctx) throws Exception {
        User user = new User();
        reader.beginObject();

        while (reader.hasNext()) {
            reader.next();
            if ("FirstName".equals(reader.name())) {
                user.setFirstName(reader.valueAsString());
            } else if ("LastName".equals(reader.name())) {
                user.setLastName(reader.valueAsString());
            } else if ("Adress".equals(reader.name())) {
                reader.beginObject();
                while (reader.hasNext()) {
                    if ("String2".equals(reader.name())) {
                        user.setCountry(reader.valueAsString());
                    } else {
                        reader.skipValue();
                    }
                }
                reader.endObject();
            } else {
                reader.skipValue();
            }
        }

        reader.endObject();
        return user;
    }

    @Override
    public void serialize(User object, ObjectWriter writer, Context ctx) throws Exception {
        // TODO Auto-generated method stub

    }
}

错误是:

com.owlike.genson.JsonBindingException: Could not deserialize to type interface java.util.List
    at com.owlike.genson.Genson.deserialize(Genson.java:384)
    ...

Caused by: com.owlike.genson.stream.JsonStreamException: Illegal character at row 0 and column 660 expected } but read '{' !
    at com.owlike.genson.stream.JsonReader.newWrongTokenException(JsonReader.java:942)
    ...

Genson在http://genson.io/Documentation/UserGuide/#custom-serde给出了一个示例,但他们读取了一个整数列表作为值,这就是为什么我在嵌套json时尝试了嵌套的原因...

如果有人对如何解决我的问题有任何想法,非常感谢!

2 个答案:

答案 0 :(得分:1)

@eugen谢谢您的回答,我在每个bean参数上方都尝试了@JsonProperty,但不幸的是它没有用。

然后出现了一个同事,我们在UserConverter中添加了

private GenericType<Map<String, String>> adressMap = new GenericType<Map<String, String>>() {};

并更改了第二个时间:

else if ("Adress".equals(reader.name())) {
   user.string2Value((ctx.genson.deserialize(adressMap, reader, ctx)).get("String2"));
}

这实际上是有效的。

答案 1 :(得分:0)

您的问题是JSON中的属性名称与代码中的名称不同。请注意,在您的JSON中,第一个字母为大写。

您可以选择以下几种方法:   -在您的JSON中重命名它们   -在代码端使用@JsonProperty("newname")或使用builder.rename(currentName, newName)重命名。   -实施自定义PropertyNameResolver,将名称解析委托给ConventionalBeanPropertyNameResolver ,然后将第一个字母更改为大写。

您可以在开始时实现自定义转换器,但是如果您需要针对每种类型进行自定义转换器,这将是很多工作。

我建议实现自定义名称解析器。