将动态 JSON 值解析为 Java 个对象
Parsing dynamic JSON values to Java objects
在我的应用程序中,我有很多具有排序和过滤功能的概览(表格)。因为不同的列可以包含不同的值类型(字符串、数字、日期、集合等),这些列的过滤器也可以带来不同的值。让我给你看几个例子(转换为 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>"(,...)]}
现在我需要在助手 class 中解析 JSON,它将构建 SQL 查询的 WHERE
子句。在 PHP 中,这不是问题,因为我可以调用 json_decode
然后简单地检查某个值是否是 array
、string
或其他任何值......但是如何做到这一点就在 Java?
到目前为止,我正在使用 Spring 的 JsonJsonParser
(我没有发现 Spring 的不同解析器之间有任何明显的区别,例如 Jackson
、Gson
和其他人)。
我正在考虑使用三个不同的构造函数创建自己的数据对象 class 或为所有三种可能性创建三个数据对象 classes,但我不知道如何处理value 在 JSON 被解析器解析后为 column_name
返回...
简单地看一下例子,它给了我三种可能性:
Map<String, String>
Map<String, Map<String, String>>
Map<String, String[]>
有什么想法或线索吗?
Jackson 的 ObjectMapper treeToValue 应该可以帮到你。
你的主要问题是你的第一个版本 JSON 与其他两个版本的结构不同。选择另外两个你可以将你的 JSON 反序列化为你所说的 Map<String, Map<String, String>
但第一个版本适合 Map.
有几个解决方案可供您使用:
- 您将 JSON 格式更改为始终匹配
Map<String, Map<String, String>
模式
- 您首先将 JSON 解析为 JsonNode,检查值的类型并将整个事物反序列化为正确的 Map 模式。
- (又快又脏)您不更改 JSON,但您
try
使用其中一个 Map 模式,捕获 JsonProcessingException,然后使用另一个 Map 模式重试
您必须在运行时检查值的类型。您可以使用 Map<String, Object>
或 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>\"}]}"
}
}
}
在我的应用程序中,我有很多具有排序和过滤功能的概览(表格)。因为不同的列可以包含不同的值类型(字符串、数字、日期、集合等),这些列的过滤器也可以带来不同的值。让我给你看几个例子(转换为 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>"(,...)]}
现在我需要在助手 class 中解析 JSON,它将构建 SQL 查询的 WHERE
子句。在 PHP 中,这不是问题,因为我可以调用 json_decode
然后简单地检查某个值是否是 array
、string
或其他任何值......但是如何做到这一点就在 Java?
到目前为止,我正在使用 Spring 的 JsonJsonParser
(我没有发现 Spring 的不同解析器之间有任何明显的区别,例如 Jackson
、Gson
和其他人)。
我正在考虑使用三个不同的构造函数创建自己的数据对象 class 或为所有三种可能性创建三个数据对象 classes,但我不知道如何处理value 在 JSON 被解析器解析后为 column_name
返回...
简单地看一下例子,它给了我三种可能性:
Map<String, String>
Map<String, Map<String, String>>
Map<String, String[]>
有什么想法或线索吗?
Jackson 的 ObjectMapper treeToValue 应该可以帮到你。
你的主要问题是你的第一个版本 JSON 与其他两个版本的结构不同。选择另外两个你可以将你的 JSON 反序列化为你所说的 Map<String, Map<String, String>
但第一个版本适合 Map.
有几个解决方案可供您使用:
- 您将 JSON 格式更改为始终匹配
Map<String, Map<String, String>
模式 - 您首先将 JSON 解析为 JsonNode,检查值的类型并将整个事物反序列化为正确的 Map 模式。
- (又快又脏)您不更改 JSON,但您
try
使用其中一个 Map 模式,捕获 JsonProcessingException,然后使用另一个 Map 模式重试
您必须在运行时检查值的类型。您可以使用 Map<String, Object>
或 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>\"}]}"
}
}
}