将动态JSON值解析为Java对象

时间:2015-09-22 15:59:36

标签: java json spring spring-boot json-deserialization

在我的应用程序中,我有很多具有排序和过滤功能的概述(表格)。并且因为不同的列可以容纳不同的值类型(字符串,数字,日期,集合等),这些列的过滤器也可以带来不同的值。让我向您展示一些示例(已经转换为JSON,因为通过REST请求发送到服务器):

对于简单的字符串值,它就像:

{"<column_name>":"<value>"}

对于数字和日期列,过滤器如下所示:

{"<column_name>":[{"operator":"eq","value":"<value>"}]}
{"<column_name>":[{"operator":"eq","value":"<value1>"},{"operator":"gt","value":"<value2>"}]}

对于设置,过滤器看起来像

{"<column_name>":["<value1>","<value2>"(,...)]}

现在我需要在一个帮助器类中解析该JSON,该类将构建SQL查询的WHERE子句。在PHP中,这不是问题,因为我可以调用json_decode然后只检查某个值是arraystring还是其他任何...但是如何在Java中执行此操作?

到目前为止,我正在使用Spring的JsonJsonParser(我发现Spring之类的不同解析器之间没有任何明显的区别,如JacksonGson和其他人。

我正在考虑使用三种不同的构造函数创建一个自己的数据对象类,或者为这三种可能性创建三个数据对象类,但我不知道如何处理返回的解析器解析JSON后的column_name ...

简单地看一下这些例子,它给了我三种可能性:

  1. Map<String, String>
  2. Map<String, Map<String, String>>
  3. Map<String, String[]>
  4. 任何想法或线索?

2 个答案:

答案 0 :(得分:3)

Jackson的ObjectMapper treeToValue应该可以帮助你。

http://fasterxml.github.io/jackson-databind/javadoc/2.2.0/com/fasterxml/jackson/databind/ObjectMapper.html#treeToValue%28com.fasterxml.jackson.core.TreeNode,%20java.lang.Class%29

您的主要问题是您的第一个版本JSON与其他两个版本的结构不同。选择其他两个,您可以将JSON反序列化为wget,如您所说,但第一个版本适合Map。

您可以使用以下几种解决方案:

  • 您将JSON格式更改为始终与Map<String, Map<String, String>模式
  • 匹配
  • 首先将JSON解析为JsonNode,检查值的类型并将整个事件反序列化为正确的Map模式。
  • (快速和脏)您不更改JSON,但是您Map<String, Map<String, String>使用其中一个Map模式,捕获JsonProcessingException,然后使用其他Map模式重试

答案 1 :(得分:1)

您必须在运行时检查值的类型。您可以使用Map<String, Object>JsonNode

Map&lt; String,Object&gt;

JsonParser parser = JsonParserFactory.getJsonParser();
Map<String, Object> map = parser.parseMap(str);
Object filterValue = filter.get("<column_name>");
if (filterValue instanceof String) {
  // str is like "{\"<column_name>\":\"<value>\"}"
} else if (filterValue instanceof Collection) {
  for (Object arrayValue : (Collection<Object>) filterValue) {
    if (arrayValue instanceof String) {
      // str is like "{\"<column_name>\":[\"<value1>\",\"<value2>\"]}"
    } else if (arrayValue instanceof Map) {
      // str is like "{\"<column_name>\":[{\"operator\":\"eq\",\"value\":\"<value>\"}]}"
    }
  }
}

JsonNode

ObjectMapper mapper = new ObjectMapper();
JsonNode filter = mapper.readTree(str);
JsonNode filterValue = filter.get("<column_name>");
if (filterValue.isTextual()) {
  // str is like "{\"<column_name>\":\"<value>\"}"
} else if (filterValue.isArray()) {
  for (JsonNode arrayValue : filterValue.elements()) {
    if (arrayValue.isTextual()) {
      // str is like "{\"<column_name>\":[\"<value1>\",\"<value2>\"]}"
    } else if (arrayValue.isObject()) {
      // str is like "{\"<column_name>\":[{\"operator\":\"eq\",\"value\":\"<value>\"}]}"
    }
  }
}