电子书 – Spring Cloud 指南 – NPI EA (分类=Spring Cloud)
announcement - icon

让我们开始使用 Spring Cloud 的微服务架构

>> 加入 Pro 并下载电子书

电子书 – Mockito – NPI EA (标签 = Mockito)
announcement - icon

模拟是单元测试的重要组成部分,Mockito 库使编写 清晰直观的单元测试 变得容易,用于您的 Java 代码。

通过我们的 Mockito 指南 开始模拟,并改进您的应用程序测试

下载电子书

电子书 – Java 并发 – NPI EA (分类=Java 并发)
announcement - icon

在应用程序中处理并发可能是一个棘手的过程,其中包含许多 潜在的陷阱。 扎实的掌握基本知识将有助于最大程度地减少这些问题。

通过我们的 Java 并发 指南开始了解多线程应用程序

>> 下载电子书

电子书 – 响应式 – NPI EA (分类=响应式)
announcement - icon

Spring 5 增加了对使用 Spring WebFlux 模块进行响应式编程的支持,此支持自那时起不断改进。 开始使用 Reactor 项目基础知识和 Spring Boot 中的响应式编程

>> 加入 Pro 并下载电子书

电子书 – Java Streams – NPI EA (分类=Java Streams)
announcement - icon

自从 Java 8 引入以来,Stream API 已成为 Java 开发的基础。 基本操作,例如迭代、过滤、映射元素序列,使用起来看似很简单。

但这些也可能被过度使用并陷入一些常见陷阱。

更好地了解 Stream 的工作方式 以及如何将其与其他语言功能结合使用,请查看我们关于 Java Streams 的指南

>> 加入 Pro 并下载电子书

电子书 – Jackson – NPI EA (分类=Jackson)
announcement - icon

用 Jackson 正确处理 JSON

下载电子书

电子书 – HTTP 客户端 – NPI EA (分类=Http 客户端)
announcement - icon

充分利用 Apache HTTP 客户端

下载电子书

电子书 – Maven – NPI EA (分类 = Maven)
announcement - icon

开始使用 Apache Maven

下载电子书

电子书 – 持久化 – NPI EA (分类=持久化)
announcement - icon

您在努力实现正确的持久化层 Spring 吗?

探索电子书

电子书 – RwS – NPI EA (分类=Spring MVC)
announcement - icon

使用 Spring 构建 REST API 吗?

下载电子书

课程 – LS – NPI EA (分类=Jackson)
announcement - icon

通过 Learn Spring 课程开始学习 Spring 和 Spring Boot

>> 学习 SPRING
课程 – RWSB – NPI EA (分类=REST)
announcement - icon

通过构建一个完整的 REST API,深入了解 Spring Boot 3 和 Spring 6,使用该框架

>> 全新的“REST With Spring Boot”

课程 – LSS – NPI EA (分类=Spring Security)
announcement - icon

是的,Spring Security 可能很复杂,从核心内的更高级功能到框架中深入的 OAuth 支持。

我将安全材料构建为 两个完整的课程 - 核心和 OAuth,以针对这些更复杂的场景进行实践。 我们探索何时以及如何使用每个功能,并 在后台项目中对其进行编码

您可以在这里探索该课程

>> 学习 Spring Security

课程 – LSD – NPI EA (标签=Spring Data JPA)
announcement - icon

Spring Data JPA 是处理 JPA 复杂性的绝佳方式,它具有 Spring Boot 的强大简洁性

通过引导式参考课程开始使用 Spring Data JPA

>> 查看课程

合作伙伴 – Moderne – NPI EA (类别=Spring Boot)
announcement - icon

使用 OpenRewrite 安全且自动地重构 Java 代码。

手动重构大型代码库既缓慢、有风险,又容易拖延。OpenRewrite 应运而生。这个用于大规模、自动化代码转换的开源框架可以帮助团队安全、一致地进行现代化改造。

每个月,OpenRewrite 的创建者和维护者 Moderne 都会举办现场、实践培训课程——一个面向初学者,一个面向经验丰富的用户。您将了解配方的运作方式、如何将其应用于项目,以及如何自信地进行代码现代化改造。

参加下一次课程,带来您的问题,并学习如何自动化通常会占用您 sprint 时间的工作。

合作伙伴 – LambdaTest – NPI EA (类别=测试)
announcement - icon

回归测试是发布流程中的重要步骤,以确保新代码不会破坏现有功能。随着代码库的不断发展,我们希望频繁运行这些测试,以便尽早发现任何问题。

确保这些测试以自动化的方式频繁运行的最佳方法当然是将其包含在 CI/CD 管道中。 这样,每次向仓库提交代码时,回归测试将自动执行。

在本教程中,我们将学习如何使用 Selenium 创建回归测试,然后使用 GitHub Actions 将它们包含在我们的管道中,在 LambdaTest 云网格上运行

>> 如何使用 GitHub Actions 运行 Selenium 回归测试

课程 – LJB – NPI EA (类别 = Core Java)
announcement - icon

通过编码方式构建 Java 的坚实、实用的基础

>> 学习 Java 基础

电子书 – Jackson – NPI (分类=Jackson)
announcement - icon

Jackson 和 JSON 在 Java 中,最终通过实践学习的方式掌握

>> 下载电子书

1. 概述

本教程将重点介绍在 Jackson 中使用 树模型节点。

我们将使用 JsonNode 进行各种转换以及添加、修改和删除节点。

2. 创建节点

创建节点的第一个步骤是使用默认构造函数实例化一个 ObjectMapper 对象

ObjectMapper mapper = new ObjectMapper();

由于创建 ObjectMapper 对象代价较高,建议我们重用同一个对象进行多次操作。

接下来,一旦我们有了 ObjectMapper,就有三种不同的方法可以创建树节点。

2.1. 从头开始构建节点

这是从无到有创建节点的最常见方法

JsonNode node = mapper.createObjectNode();

或者,我们也可以通过 JsonNodeFactory 创建节点

JsonNode node = JsonNodeFactory.instance.objectNode();

2.2. 从 JSON 来源解析

此方法在 Jackson – Marshall String to JsonNode 文章中得到了很好的介绍。请参阅它以获取更多信息。

2.3. 从对象转换

可以通过调用 ObjectMapper 上的 valueToTree(Object fromValue) 方法将节点从 Java 对象转换而来

JsonNode node = mapper.valueToTree(fromValue);

convertValue API 在这里也很有帮助

JsonNode node = mapper.convertValue(fromValue, JsonNode.class);

让我们看看它在实践中是如何工作的。

假设我们有一个名为 NodeBean 的类

public class NodeBean {
    private int id;
    private String name;

    public NodeBean() {
    }

    public NodeBean(int id, String name) {
        this.id = id;
        this.name = name;
    }

    // standard getters and setters
}

让我们编写一个测试来确保转换发生正确

@Test
public void givenAnObject_whenConvertingIntoNode_thenCorrect() {
    NodeBean fromValue = new NodeBean(2016, "baeldung.com");

    JsonNode node = mapper.valueToTree(fromValue);

    assertEquals(2016, node.get("id").intValue());
    assertEquals("baeldung.com", node.get("name").textValue());
}

3. 转换节点

3.1. 写入为 JSON

这是将树节点转换为 JSON 字符串的基本方法,其中目标可以是 FileOutputStreamWriter

mapper.writeValue(destination, node);

通过重用第 2.3 节中声明的类 NodeBean,一个测试确保此方法按预期工作

final String pathToTestFile = "node_to_json_test.json";

@Test
public void givenANode_whenModifyingIt_thenCorrect() throws IOException {
    String newString = "{\"nick\": \"cowtowncoder\"}";
    JsonNode newNode = mapper.readTree(newString);

    JsonNode rootNode = ExampleStructure.getExampleRoot();
    ((ObjectNode) rootNode).set("name", newNode);

    assertFalse(rootNode.path("name").path("nick").isMissingNode());
    assertEquals("cowtowncoder", rootNode.path("name").path("nick").textValue());
}

3.2. 转换为对象

JsonNode 转换为 Java 对象的最佳方法是 treeToValue API

NodeBean toValue = mapper.treeToValue(node, NodeBean.class);

这在功能上等同于以下代码

NodeBean toValue = mapper.convertValue(node, NodeBean.class)

我们也可以通过令牌流来完成

JsonParser parser = mapper.treeAsTokens(node);
NodeBean toValue = mapper.readValue(parser, NodeBean.class);

最后,让我们实现一个测试来验证转换过程

@Test
public void givenANode_whenConvertingIntoAnObject_thenCorrect()
  throws JsonProcessingException {
    JsonNode node = mapper.createObjectNode();
    ((ObjectNode) node).put("id", 2016);
    ((ObjectNode) node).put("name", "baeldung.com");

    NodeBean toValue = mapper.treeToValue(node, NodeBean.class);

    assertEquals(2016, toValue.getId());
    assertEquals("baeldung.com", toValue.getName());
}

4. 操作树节点

我们将使用以下 JSON 元素,包含在一个名为 example.json 的文件中,作为要执行操作的基础结构

{
    "name": 
        {
            "first": "Tatu",
            "last": "Saloranta"
        },

    "title": "Jackson founder",
    "company": "FasterXML"
}

此 JSON 文件位于类路径上,被解析为一个模型树

public class ExampleStructure {
    private static ObjectMapper mapper = new ObjectMapper();

    static JsonNode getExampleRoot() throws IOException {
        InputStream exampleInput = 
          ExampleStructure.class.getClassLoader()
          .getResourceAsStream("example.json");
        
        JsonNode rootNode = mapper.readTree(exampleInput);
        return rootNode;
    }
}

请注意,树的根将在以下子节中说明节点操作时使用。

4.1. 查找节点

在使用任何节点之前,我们需要做的第一件事是找到它并将其分配给一个变量。

如果我们事先知道节点的路径,那么很容易做到。

假设我们想要一个名为 last 的节点,它位于 name 节点之下

JsonNode locatedNode = rootNode.path("name").path("last");

或者,也可以使用 getwith API 代替 path

如果路径未知,搜索当然会变得更复杂和迭代。

我们可以在 第 5 节 – 迭代节点 中看到遍历所有节点的示例。

4.2. 添加新节点

可以将节点作为另一个节点的子节点添加

ObjectNode newNode = ((ObjectNode) locatedNode).put(fieldName, value);

可以使用许多重载的 put 变体来添加不同值类型的新节点。

还有许多其他类似的方法可用,包括 putArrayputObjectPutPOJOputRawValueputNull

最后,让我们看一个示例,我们将整个结构添加到树的根节点

"address":
{
    "city": "Seattle",
    "state": "Washington",
    "country": "United States"
}

这是完整的测试,经过所有这些操作并验证结果

@Test
public void givenANode_whenAddingIntoATree_thenCorrect() throws IOException {
    JsonNode rootNode = ExampleStructure.getExampleRoot();
    ObjectNode addedNode = ((ObjectNode) rootNode).putObject("address");
    addedNode
      .put("city", "Seattle")
      .put("state", "Washington")
      .put("country", "United States");

    assertFalse(rootNode.path("address").isMissingNode());
    
    assertEquals("Seattle", rootNode.path("address").path("city").textValue());
    assertEquals("Washington", rootNode.path("address").path("state").textValue());
    assertEquals(
      "United States", rootNode.path("address").path("country").textValue();
}

4.3. 编辑节点

可以通过调用 set(String fieldName, JsonNode value) 方法修改 ObjectNode 实例

JsonNode locatedNode = locatedNode.set(fieldName, value);

通过使用相同类型的对象上的 replacesetAll 方法可以获得类似的结果。

为了验证该方法是否按预期工作,我们将更改根节点下字段 name 的值,从包含 firstlast 对象的结构更改为仅包含 nick 字段的结构,并在测试中进行验证。

@Test
public void givenANode_whenModifyingIt_thenCorrect() throws IOException {
    String newString = "{\"nick\": \"cowtowncoder\"}";
    JsonNode newNode = mapper.readTree(newString);

    JsonNode rootNode = ExampleStructure.getExampleRoot();
    ((ObjectNode) rootNode).set("name", newNode);

    assertFalse(rootNode.path("name").path("nick").isMissingNode());
    assertEquals("cowtowncoder", rootNode.path("name").path("nick").textValue());
}

4.4. 移除节点

可以通过调用其父节点上的 remove(String fieldName) API 来移除节点。

JsonNode removedNode = locatedNode.remove(fieldName);

为了同时移除多个节点,我们可以调用一个重载的方法,该方法参数类型为 Collection<String>,它返回父节点而不是要移除的节点。

ObjectNode locatedNode = locatedNode.remove(fieldNames);

在想要删除给定节点的所有子节点这种极端情况下,removeAll API 会派上用场。

以下测试将重点关注上述第一种方法,这是最常见的情况。

@Test
public void givenANode_whenRemovingFromATree_thenCorrect() throws IOException {
    JsonNode rootNode = ExampleStructure.getExampleRoot();
    ((ObjectNode) rootNode).remove("company");

    assertTrue(rootNode.path("company").isMissingNode());
}

5. 遍历节点

让我们遍历 JSON 文档中的所有节点,并将它们重新格式化为 YAML。

JSON 有三种类型的节点,分别是 Value(值)、Object(对象)和 Array(数组)。

因此,让我们通过添加一个 Array 来确保我们的示例数据包含所有三种不同的类型。

{
    "name": 
        {
            "first": "Tatu",
            "last": "Saloranta"
        },

    "title": "Jackson founder",
    "company": "FasterXML",
    "pets" : [
        {
            "type": "dog",
            "number": 1
        },
        {
            "type": "fish",
            "number": 50
        }
    ]
}

现在让我们看看我们想要生成的 YAML。

name: 
  first: Tatu
  last: Saloranta
title: Jackson founder
company: FasterXML
pets: 
- type: dog
  number: 1
- type: fish
  number: 50

我们知道 JSON 节点具有分层树结构。 因此,遍历整个 JSON 文档最简单的方法是从顶部开始,然后通过所有子节点向下工作。

我们将把根节点传递给一个递归方法。 然后,该方法将使用提供的节点的每个子节点调用自身。

5.1. 测试遍历

我们将首先创建一个简单的测试,以检查我们是否可以成功地将 JSON 转换为 YAML。

我们的测试将 JSON 文档的根节点提供给我们的 toYaml 方法,并断言返回的值是我们期望的值。

@Test
public void givenANodeTree_whenIteratingSubNodes_thenWeFindExpected() throws IOException {
    JsonNode rootNode = ExampleStructure.getExampleRoot();
    
    String yaml = onTest.toYaml(rootNode);

    assertEquals(expectedYaml, yaml); 
}

public String toYaml(JsonNode root) {
    StringBuilder yaml = new StringBuilder(); 
    processNode(root, yaml, 0); 
    return yaml.toString(); }
}

5.2. 处理不同类型的节点

我们需要略微不同地处理不同类型的节点。

我们将在我们的 processNode 方法中执行此操作。

private void processNode(JsonNode jsonNode, StringBuilder yaml, int depth) {
    if (jsonNode.isValueNode()) {
        yaml.append(jsonNode.asText());
    }
    else if (jsonNode.isArray()) {
        for (JsonNode arrayItem : jsonNode) {
            appendNodeToYaml(arrayItem, yaml, depth, true);
        }
    }
    else if (jsonNode.isObject()) {
        appendNodeToYaml(jsonNode, yaml, depth, false);
    }
}

首先,让我们考虑一个 Value 节点。 我们只需调用节点的 asText 方法来获取值的 String 表示形式。

接下来,让我们看一下 Array 节点。 Array 节点内的每个项目本身都是一个 JsonNode,因此我们遍历 Array 并将每个节点传递给 appendNodeToYaml 方法。 我们还需要知道这些节点是数组的一部分。

不幸的是,节点本身不包含任何表明这一点的内容,因此我们将一个标志传递给我们的 appendNodeToYaml 方法。

最后,我们想要遍历每个 Object 节点的子节点。 一种选择是使用 JsonNode.elements

但是,我们无法从元素确定字段名,因为它只包含字段值。

Object  {"first": "Tatu", "last": "Saloranta"}
Value  "Jackson Founder"
Value  "FasterXML"
Array  [{"type": "dog", "number": 1},{"type": "fish", "number": 50}]

相反,我们将使用 JsonNode.fields,因为它使我们可以访问字段名和值。

Key="name", Value=Object  {"first": "Tatu", "last": "Saloranta"}
Key="title", Value=Value  "Jackson Founder"
Key="company", Value=Value  "FasterXML"
Key="pets", Value=Array  [{"type": "dog", "number": 1},{"type": "fish", "number": 50}]

对于每个字段,我们将字段名添加到输出中,然后通过将其传递给 processNode 方法来处理该值作为子节点。

private void appendNodeToYaml(
  JsonNode node, StringBuilder yaml, int depth, boolean isArrayItem) {
    Iterator<Entry<String, JsonNode>> fields = node.fields();
    boolean isFirst = true;
    while (fields.hasNext()) {
        Entry<String, JsonNode> jsonField = fields.next();
        addFieldNameToYaml(yaml, jsonField.getKey(), depth, isArrayItem && isFirst);
        processNode(jsonField.getValue(), yaml, depth+1);
        isFirst = false;
    }
        
}

我们无法从节点判断它有多少祖先。

因此,我们将一个名为 depth 的字段传递给 processNode 方法来跟踪这一点,并且每次获得子节点时都会增加此值,以便我们可以正确地缩进 YAML 输出中的字段。

private void addFieldNameToYaml(
  StringBuilder yaml, String fieldName, int depth, boolean isFirstInArray) {
    if (yaml.length()>0) {
        yaml.append("\n");
        int requiredDepth = (isFirstInArray) ? depth-1 : depth;
        for(int i = 0; i < requiredDepth; i++) {
            yaml.append("  ");
        }
        if (isFirstInArray) {
            yaml.append("- ");
        }
    }
    yaml.append(fieldName);
    yaml.append(": ");
}

现在我们已经准备好所有代码来遍历节点并生成 YAML 输出,我们可以运行我们的测试来证明它有效。

6. 结论

本文涵盖了在使用 Jackson 处理树模型时的常用 API 和场景。

支持本文的代码可在 GitHub 上获取。 一旦你Baeldung Pro 会员 身份登录,就开始学习并在项目上进行编码。
Baeldung Pro – NPI EA (类别 = Baeldung)
announcement - icon

Baeldung Pro 具有完全无广告以及最终具有深色模式,提供干净的学习体验

>> 探索干净的 Baeldung

一旦早期采用者的席位全部用完,价格将上涨并保持在每年 33 美元。

电子书 – HTTP 客户端 – NPI EA (类别=HTTP 客户端)
announcement - icon

Apache HTTP Client 是一个非常强大的库,适用于简单和高级用例,在测试 HTTP 端点时尤其适用。 查看我们的指南,涵盖基本请求和响应处理,以及安全性、Cookie、超时等。

>> 下载电子书

电子书 – Java 并发 – NPI EA (分类=Java 并发)
announcement - icon

在应用程序中处理并发可能是一个棘手的过程,其中包含许多 潜在的陷阱。 扎实的掌握基本知识将有助于最大程度地减少这些问题。

通过我们的 Java 并发 指南开始了解多线程应用程序

>> 下载电子书

电子书 – Java Streams – NPI EA (分类=Java Streams)
announcement - icon

自从 Java 8 引入以来,Stream API 已成为 Java 开发的基础。 基本操作,例如迭代、过滤、映射元素序列,使用起来看似很简单。

但这些也可能被过度使用并陷入一些常见陷阱。

更好地了解 Stream 的工作方式 以及如何将其与其他语言功能结合使用,请查看我们关于 Java Streams 的指南

>> 加入 Pro 并下载电子书

电子书 – 持久化 – NPI EA (分类=持久化)
announcement - icon

您在努力实现正确的持久化层 Spring 吗?

探索电子书

课程 – LS – NPI EA (类别=REST)

announcement - icon

从 Spring Boot 开始,通过 Learn Spring 课程了解核心 Spring。

>> 查看课程

合作伙伴 – Moderne – NPI EA (标签=重构)
announcement - icon

现代 Java 团队行动迅速——但代码库并不总是跟上。 框架会发生变化,依赖关系会漂移,技术债务会累积,直到它开始拖慢交付速度。 OpenRewrite 就是为此而构建的:一个开源重构引擎,可在保持开发人员意图不变的同时自动化重复的代码更改。

由 Moderne 的 OpenRewrite 创建者和维护者领导的每月培训系列,将介绍实际的迁移和现代化模式。 无论您是重构配方的新手,还是准备编写自己的配方,您都将学习以安全且可扩展的方式进行重构的实用方法。

如果您曾经希望重构感觉像编写代码一样自然——并且一样快速——这是一个很好的起点

电子书 Jackson – NPI EA – 3 (类别 = Jackson)
电子书 Jackson – NPI (cat = Jackson)
© .