【问题标题】:Parsing dynamic JSON values to Java objects将动态 JSON 值解析为 Java 对象
【发布时间】:2015-09-22 15:59:36
【问题描述】:

在我的应用程序中,我有很多带有排序和过滤功能的概览(表格)。并且因为不同的列可以保存不同的值类型(字符串、数字、日期、集合等),这些列的过滤器也可以带来不同的值。让我向您展示几个示例(已转换为 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 等)。

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

只看例子,它给了我三种可能性:

  1. Map&lt;String, String&gt;
  2. Map&lt;String, Map&lt;String, String&gt;&gt;
  3. Map&lt;String, String[]&gt;

有什么想法或线索吗?

【问题讨论】:

  • 你能解释一下JsonJsonParser是什么吗?
  • @SotiriosDelimanolis 它是 org.springframework.boot.json 包的一部分,由以下解析器组成:BasicGsonJacksonJsonJsonSimpleYaml(后缀为JsonParser 获取全名,因此您将获得例如 JsonJsonParser)。

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


【解决方案1】:

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 反序列化为 Map&lt;String, Map&lt;String, String&gt;,但第一个版本适合 Map。

有几种解决方案可供您使用:

  • 您将 JSON 格式更改为始终匹配 Map&lt;String, Map&lt;String, String&gt; 模式
  • 您首先将 JSON 解析为 JsonNode,检查值的类型并将整个内容反序列化为正确的 Map 模式。
  • (又快又脏)您不更改 JSON,但您 try 使用其中一种 Map 模式,捕获 JsonProcessingException,然后使用另一种 Map 模式重试

【讨论】:

  • 我建议放弃Map,只使用JsonNode 及其子类型。
【解决方案2】:

您必须在运行时检查值的类型。您可以使用Map&lt;String, Object&gt;JsonNode

地图

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>\"}]}"
    }
  }
}

【讨论】:

  • 谢谢,我喜欢你的回答,因为它带来了代码示例。看着它们,我可以看到使用 Map&lt;String, Object&gt;JsonNode 之间并没有太大的区别,是吗?我想我会选择JsonNode,因为在我看来更好...
  • 是的,我更喜欢JsonNode,与Map&lt;String, Object&gt; 一样,您可以获得更高级别的 JSON 数据表示...
猜你喜欢
  • 2022-08-20
  • 2014-10-17
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2018-01-09
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多