java - 如何使用jsonPath选择jsonfile不同级别的字段?
问题描述
我想将 jsonobjcts 转换为 csv 文件。到目前为止,Wy(工作)尝试是将 json 文件加载为 JSONObject(来自 googlecode.josn-simple 库),然后使用 jsonPath 将它们转换为字符串数组,然后用于构建 csv 行。但是我遇到了 jsonPath 的问题。从给定的示例 json...
{
"issues": [
{
"key": "abc",
"fields": {
"issuetype": {
"name": "Bug",
"id": "1",
"subtask": false
},
"priority": {
"name": "Major",
"id": "3"
},
"created": "2020-5-11",
"status": {
"name": "OPEN"
}
}
},
{
"key": "def",
"fields": {
"issuetype": {
"name": "Info",
"id": "5",
"subtask": false
},
"priority": {
"name": "Minor",
"id": "2"
},
"created": "2020-5-8",
"status": {
"name": "DONE"
}
}
}
]}
我想选择以下内容:
[
"abc",
"Bug",
"Major",
"2020-5-11",
"OPEN",
"def",
"Info",
"Minor",
"2020-5-8",
"DONE"
]
csv 应如下所示:
abc,Bug,Major,2020-5-11,OPEN
def,Info,Minor,2020-5-8,DONE
我试过了$.issues.[*].[key,fields]
,我得到了
"abc",
{
"issuetype": {
"name": "Bug",
"id": "1",
"subtask": false
},
"priority": {
"name": "Major",
"id": "3"
},
"created": "2020-5-11",
"status": {
"name": "OPEN"
}
},
"def",
{
"issuetype": {
"name": "Info",
"id": "5",
"subtask": false
},
"priority": {
"name": "Minor",
"id": "2"
},
"created": "2020-5-8",
"status": {
"name": "DONE"
}
}
]
但是当我想选择例如“创建”时$.issues.[*].[key,fields.[created]
[
"2020-5-11",
"2020-5-8"
]
这就是结果。
但我只是不知道如何在字段问题类型中选择“键”和例如“名称”。我如何使用 jsonPath 来做到这一点,或者是否有更好的方法来过滤 jsonfile 然后将其转换为 csv?
解决方案
我推荐我认为更好的方法——即创建一组代表 JSON 数据结构的 Java 类。当您将 JSON 读入这些类时,您可以使用标准 Java 操作数据。
我还推荐了一个不同的 JSON 解析器——在这个例子中是杰克逊,但还有其他的。为什么?主要是熟悉度——有关更多说明,请参见稍后。
从最终结果开始:假设我有一个名为的类Container
,其中包含 JSON 文件中列出的所有问题,然后我可以使用以下内容填充它:
//import com.fasterxml.jackson.databind.ObjectMapper;
String jsonString = "{...}" // your JSON data as a string, for this demo.
ObjectMapper objectMapper = new ObjectMapper();
Container container = objectMapper.readValue(jsonString, Container.class);
现在我可以打印出你想要的 CSV 格式的所有问题,如下所示:
container.getIssues().forEach((issue) -> {
printCsvRow(issue);
});
在这里,该printCsvRow()
方法如下所示:
private void printCsvRow(Issue issue) {
String key = issue.getKey();
Fields fields = issue.getFields();
String type = fields.getIssuetype().getName();
String priority = fields.getPriority().getName();
String created = fields.getCreated();
String status = fields.getStatus().getName();
System.out.println(String.join(",", key, type, priority, created, status));
}
实际上,我会使用 CSV 库来确保记录的格式正确——以上只是为了说明,展示如何访问 JSON 数据。
打印以下内容:
abc,Bug,Major,2020-5-11,OPEN
def,Info,Minor,2020-5-8,DONE
并且只过滤 OPEN 记录,我可以这样做:
container.getIssues()
.stream()
.filter(issue -> issue.getFields().getStatus().getName().equals("OPEN"))
.forEach((issue) -> {
printCsvRow(issue);
});
打印以下内容:
abc,Bug,Major,2020-5-11,OPEN
为了启用 Jackson,我使用具有以下依赖项的 Maven:
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.10.3</version>
</dependency>
如果您不使用 Maven,这会给我 3 个 JAR:jackson-databind
、jackson-annotations
和jackson-core
.
为了创建我需要的嵌套 Java 类(以反映 JSON 的结构),我使用了一个工具,该工具使用您的示例 JSON 为我生成它们。
就我而言,我使用了这个工具,但还有其他工具。
我选择“Container”作为根 Java 类的名称;JSON 的源类型;并选择了 Jackson 2.x 注释。我还要求 getter 和 setter。
我将生成的类(字段、问题、问题类型、优先级、状态和容器)添加到我的项目中。
警告:这些 Java 类的完整性仅与示例 JSON 一样好。当然,您可以增强这些类以更准确地反映您需要处理的实际 JSON。
JacksonObjectMapper
负责将 JSON 加载到类结构中。
我选择使用 Jackson 而不是 JsonPath,只是因为熟悉。JsonPath 似乎具有非常相似的对象映射功能——但我从未使用过 JsonPath 的这些功能。
最后说明:您可以在 JsonPath 中使用 xpath 样式谓词来访问单个数据项和项组 - 正如您在问题中所描述的那样。但是(根据我的经验)如果您想以更灵活的方式处理所有数据,那么创建 Java 类几乎总是值得付出额外努力——尤其是当这涉及将 JSON 输入转换为不同的输出结构时。
推荐阅读
- java - 重定向到带有连字符的 URL 在 Spring Gateway 应用程序中不起作用
- python - Pytorch CNN 损失没有改变
- excel - 具有动态范围/边界的循环
- python - 使用 CrossHair 区分自动机和 python 函数
- csh - 如何设置我的 tcsh 提示符以显示上一个命令的执行时间?
- node.js - Rails 6 - 我的应用程序中的 node_modules 文件夹是什么?
- reactjs - React:如何打开卡片的特定编辑/删除窗口?
- javascript - 无法离开会话,本地媒体已在 Safari 中停止 - IceLink 3
- python - TensorFlow Serving 返回 400 Bad Request 错误
- python - 如何根据升序值过滤列表?