电子书 – 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. 概述

在本快速教程中,我们将学习如何控制 Java 枚举在使用 Jackson 2 时进行序列化和反序列化的方式

要深入了解并学习 Jackson 2 的其他很酷的功能,请访问 主要的 Jackson 教程

2. 控制枚举的表示形式

让我们定义以下枚举

public enum Distance {
    KILOMETER("km", 1000), 
    MILE("miles", 1609.34),
    METER("meters", 1), 
    INCH("inches", 0.0254),
    CENTIMETER("cm", 0.01), 
    MILLIMETER("mm", 0.001);

    private String unit;
    private final double meters;

    private Distance(String unit, double meters) {
        this.unit = unit;
        this.meters = meters;
    }

    // standard getters and setters
}

3. 将枚举序列化为 JSON

3.1. 默认枚举表示形式

默认情况下,Jackson 会将 Java 枚举表示为简单的字符串。例如

new ObjectMapper().writeValueAsString(Distance.MILE);

将会得到

"MILE"

但是,在将此 枚举编组到 JSON 对象时,我们希望得到类似以下的内容

{"unit":"miles","meters":1609.34}

3.2. 枚举作为 JSON 对象

从 Jackson 2.1.2 开始,现在有一个配置选项可以处理这种表示形式。这可以通过在类级别使用 @JsonFormat 注释来完成

@JsonFormat(shape = JsonFormat.Shape.OBJECT)
public enum Distance { ... }

这将导致在序列化此 枚举Distance.MILE 时得到期望的结果

{"unit":"miles","meters":1609.34}

3.3. 枚举和 @JsonValue

控制枚举编组输出的另一种简单方法是在 getter 上使用 @JsonValue 注释

public enum Distance { 
    ...
 
    @JsonValue
    public String getMeters() {
        return meters;
    }
}

我们表达的是 getMeters() 是此枚举的实际表示形式。因此,序列化结果将是

1609.34

3.4. 枚举的自定义序列化器

如果我们使用的是 Jackson 2.1.2 之前的版本,或者需要对枚举进行更多自定义,我们可以使用一个 自定义 Jackson 序列化器。 首先,我们需要定义它

public class DistanceSerializer extends StdSerializer {
    
    public DistanceSerializer() {
        super(Distance.class);
    }

    public DistanceSerializer(Class t) {
        super(t);
    }

    public void serialize(
      Distance distance, JsonGenerator generator, SerializerProvider provider) 
      throws IOException, JsonProcessingException {
        generator.writeStartObject();
        generator.writeFieldName("name");
        generator.writeString(distance.name());
        generator.writeFieldName("unit");
        generator.writeString(distance.getUnit());
        generator.writeFieldName("meters");
        generator.writeNumber(distance.getMeters());
        generator.writeEndObject();
    }
}

然后我们可以将序列化器应用于将被序列化的类

@JsonSerialize(using = DistanceSerializer.class)
public enum TypeEnum { ... }

结果如下

{"name":"MILE","unit":"miles","meters":1609.34}

4. 将 JSON 反序列化为枚举

首先,让我们定义一个具有 Distance 成员的City

public class City {
    
    private Distance distance;
    ...    
}

然后我们将讨论将 JSON 字符串反序列化为枚举的不同方法。

4.1. 默认行为

默认情况下,Jackson 将使用枚举名称从 JSON 反序列化.

例如,它将反序列化 JSON

{"distance":"KILOMETER"}

Distance.KILOMETER 对象

City city = new ObjectMapper().readValue(json, City.class);
assertEquals(Distance.KILOMETER, city.getDistance());

如果希望 Jackson 通过枚举名称以不区分大小写的方式从 JSON 反序列化,我们需要 自定义 ObjectMapper 以启用 ACCEPT_CASE_INSENSITIVE_ENUMS 功能

假设我们有另一个 JSON

{"distance":"KiLoMeTeR"}

现在,让我们进行不区分大小写的反序列化

ObjectMapper objectMapper = JsonMapper.builder()
  .enable(MapperFeature.ACCEPT_CASE_INSENSITIVE_ENUMS)
  .build();
City city = objectMapper.readValue(json, City.class);
                                                     
assertEquals(Distance.KILOMETER, city.getDistance());

如上述测试所示,我们使用 JsonMapper 构建器启用了 ACCEPT_CASE_INSENSITIVE_ENUMS 功能。

4.2. 使用 @JsonValue

我们学习了如何使用 @JsonValue 序列化枚举。我们也可以将相同的注释用于反序列化。这是可能的,因为枚举值是常量。

首先,让我们使用 @JsonValue 和其中一个 getter 方法 getMeters()

public enum Distance {
    ...

    @JsonValue
    public double getMeters() {
        return meters;
    }
}

getMeters() 方法的返回值表示枚举对象。因此,在反序列化示例 JSON

{"distance":"0.0254"}

Jackson 将查找getMeters() 返回值为 0.0254 的枚举对象。在这种情况下,该对象是 Distance.INCH

assertEquals(Distance.INCH, city.getDistance());

4.3. 使用 @JsonProperty

@JsonProperty 注释用于枚举实例

public enum Distance {
    @JsonProperty("distance-in-km")
    KILOMETER("km", 1000), 
    @JsonProperty("distance-in-miles")
    MILE("miles", 1609.34);
 
    ...
}

通过使用此注释,我们只是告诉 Jackson 将 @JsonProperty 的值映射到用此值注释的对象

由于上述声明,示例 JSON 字符串

{"distance": "distance-in-km"}

将被映射到Distance.KILOMETER 对象

assertEquals(Distance.KILOMETER, city.getDistance());

4.4. 使用 @JsonCreator

Jackson 调用带有 @JsonCreator 注解的方法来获取包含类的实例。

考虑以下 JSON 表现形式

{
    "distance": {
        "unit":"miles", 
        "meters":1609.34
    }
}

然后我们将定义带有 @JsonCreator 注解的 forValues() 工厂方法

public enum Distance {
   
    @JsonCreator
    public static Distance forValues(@JsonProperty("unit") String unit,
      @JsonProperty("meters") double meters) {
        for (Distance distance : Distance.values()) {
            if (
              distance.unit.equals(unit) && Double.compare(distance.meters, meters) == 0) {
                return distance;
            }
        }

        return null;
    }

    ...
}

请注意 @JsonProperty 注解的使用,它将 JSON 字段与方法参数绑定。

然后,当我们反序列化 JSON 示例时,我们将得到结果

assertEquals(Distance.MILE, city.getDistance());

4.5. 使用自定义 反序列化器

如果上述技术都不可用,我们可以使用自定义反序列化器。例如,我们可能无法访问 Enum 的源代码,或者我们可能 正在 使用较旧的 Jackson 版本,该版本不支持到目前为止涵盖的一个或多个注解。

根据 我们的自定义反序列化文章,为了反序列化上一节提供的 JSON,我们将首先创建反序列化类

public class CustomEnumDeserializer extends StdDeserializer<Distance> {

    @Override
    public Distance deserialize(JsonParser jsonParser, DeserializationContext ctxt)
      throws IOException, JsonProcessingException {
        JsonNode node = jsonParser.getCodec().readTree(jsonParser);

        String unit = node.get("unit").asText();
        double meters = node.get("meters").asDouble();

        for (Distance distance : Distance.values()) {
           
            if (distance.getUnit().equals(unit) && Double.compare(
              distance.getMeters(), meters) == 0) {
                return distance;
            }
        }

        return null;
    }
}

然后我们将使用 @JsonDeserialize 注解在 Enum 上指定我们自定义的反序列化器

@JsonDeserialize(using = CustomEnumDeserializer.class)
public enum Distance {
   ...
}

我们的结果是

assertEquals(Distance.MILE, city.getDistance());

5. 结论

本文说明了如何更好地控制 Java Enum 的序列化和反序列化过程和格式

支持本文的代码可在 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)
© .