0

0

处理JSON时间戳:将Epoch毫秒转换为java.time类型的策略

聖光之護

聖光之護

发布时间:2025-09-29 13:26:01

|

666人浏览过

|

来源于php中文网

原创

处理JSON时间戳:将Epoch毫秒转换为java.time类型的策略

本文旨在探讨在Java应用中,如何将REST服务返回的Epoch毫秒时间戳有效反序列化为java.time包下的LocalDateTime或LocalDate类型。我们将介绍三种主要策略:通过构造函数手动解析、配置全局Jackson反序列化规则以及实现自定义反序列化器,以解决直接转换时遇到的类型不匹配和时区信息缺失问题。

在现代java应用中,处理日期和时间数据通常推荐使用java.time包(jsr 310)中的类型,如localdatetime、localdate和instant,它们提供了更强大、更清晰的api。然而,当从外部rest服务接收json数据时,时间戳常以epoch毫秒(自1970年1月1日00:00:00 utc以来的毫秒数)的形式表示。直接尝试将这种长整型时间戳反序列化为localdatetime或localdate时,jackson库会抛出错误,例如“raw timestamp (...) not allowed for java.time.localdatetime: need additional information such as an offset or time-zone”或“invalid value for epochday”。这是因为localdatetime不包含时区信息,而epoch毫秒是一个带有隐含时区(utc)的绝对时间点。localdate则更进一步,仅表示日期,直接从epoch毫秒转换需要精确的时区和日期计算。

本文将详细介绍几种解决此问题的有效策略。

1. 通过构造函数手动解析时间戳

一种直接的方法是在目标数据类的构造函数中手动处理时间戳的转换。这种方法允许对单个类进行精确控制,适用于不希望引入全局配置或自定义反序列化器的场景。

实现步骤:

  1. 在目标数据类中定义一个接收long类型时间戳的构造函数。
  2. 使用@JsonProperty注解将JSON中的时间戳字段映射到构造函数的long参数。
  3. 在构造函数内部,利用Instant.ofEpochMilli()将Epoch毫秒转换为Instant,然后通过atZone(ZoneOffset.UTC).toLocalDateTime()转换为LocalDateTime。

示例代码:

立即学习Java免费学习笔记(深入)”;

假设我们希望将JSON中的creation_date字段(Epoch毫秒)反序列化到MyLocalApplicationClass的LocalDateTime creationDate字段。

import com.fasterxml.jackson.annotation.JsonProperty;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneOffset;

public class MyLocalApplicationClass {
    private String name;
    private LocalDateTime creationDate;
    private String createdBy;

    // 无参构造函数(或其他构造函数),如果需要的话
    public MyLocalApplicationClass() {
    }

    // 带有@JsonProperty注解的全参数构造函数,用于Jackson反序列化
    public MyLocalApplicationClass(@JsonProperty("name") String name,
                                   @JsonProperty("creation_date") long creationDate,
                                   @JsonProperty("created_by") String createdBy) {
        this.name = name;
        this.createdBy = createdBy;
        // 将Epoch毫秒转换为LocalDateTime,假设时间戳是UTC时间
        this.creationDate = Instant
            .ofEpochMilli(creationDate)
            .atZone(ZoneOffset.UTC) // 明确指定时区,通常是UTC
            .toLocalDateTime();
    }

    // Getters and Setters (省略)
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
    public LocalDateTime getCreationDate() { return creationDate; }
    public void setCreationDate(LocalDateTime creationDate) { this.creationDate = creationDate; }
    public String getCreatedBy() { return createdBy; }
    public void setCreatedBy(String createdBy) { this.createdBy = createdBy; }
}

注意事项:

  • 此方法要求在每个需要处理时间戳的类中编写类似的构造函数。
  • ZoneOffset.UTC的选择取决于你的时间戳实际代表的时区。如果时间戳是基于其他时区生成的,你需要相应调整。

2. 配置全局Jackson反序列化规则

对于Spring Boot应用,可以通过配置ObjectMapper实现全局的时间戳处理。这种方法更具侵入性,但一旦配置,所有符合条件的字段都会自动处理,减少了重复代码。

实现步骤:

  1. 引入jackson-datatype-jsr310模块: 确保你的项目中包含了Jackson对Java 8日期时间类型的支持模块。
    
        com.fasterxml.jackson.datatype
        jackson-datatype-jsr310
        2.x.x 
    
  2. 修改字段类型为Instant: 为了利用Jackson的内置功能,将目标数据类中的LocalDateTime字段类型更改为Instant。Instant是表示时间线上一个精确点的类型,与Epoch毫秒天然匹配。
  3. 配置ObjectMapper:
    • 注册JavaTimeModule。
    • 设置DeserializationFeature.READ_DATE_TIMESTAMPS_AS_NANOSECONDS为false,指示Jackson将时间戳视为毫秒而不是纳秒。

示例代码:

立即学习Java免费学习笔记(深入)”;

首先,修改MyLocalApplicationClass,将creationDate的类型改为Instant:

import com.fasterxml.jackson.annotation.JsonProperty;
import java.time.Instant;

public class MyLocalApplicationClass {
    private String name;
    @JsonProperty("creation_date") // 如果JSON字段名与Java字段名不匹配,仍需此注解
    private Instant creationDate;
    @JsonProperty("created_by")
    private String createdBy;

    // Getters and Setters (省略)
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
    public Instant getCreationDate() { return creationDate; }
    public void setCreationDate(Instant creationDate) { this.creationDate = creationDate; }
    public String getCreatedBy() { return createdBy; }
    public void setCreatedBy(String createdBy) { this.createdBy = createdBy; }
}

然后,配置ObjectMapper。

方法一:通过配置类(适用于非Spring Boot或需要更细粒度控制的场景)

超能文献
超能文献

超能文献是一款革命性的AI驱动医学文献搜索引擎。

下载
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;

@Configuration
public class JsonConfig {

    @Bean
    public Jackson2ObjectMapperBuilder jackson2ObjectMapperBuilder() {
        return new Jackson2ObjectMapperBuilder();
    }

    @Bean
    public ObjectMapper objectMapper() {
        return jackson2ObjectMapperBuilder()
            .build()
            .registerModule(new JavaTimeModule()) // 注册JavaTime模块
            .configure(DeserializationFeature.READ_DATE_TIMESTAMPS_AS_NANOSECONDS, false); // 将时间戳视为毫秒
    }
}

方法二:通过application.properties(Spring Boot推荐)

在Spring Boot应用中,JacksonAutoConfiguration会自动检测并注册JavaTimeModule。我们只需通过配置文件来设置DeserializationFeature。

# application.properties 或 application.yml
spring.jackson.deserialization.read-date-timestamps-as-nanoseconds=false

使用这种方式,你无需定义上述的JsonConfig类中的ObjectMapper和Jackson2ObjectMapperBuilder Bean。Spring Boot会自动为你配置。

注意事项:

  • 使用Instant类型后,如果需要LocalDateTime,可以在获取Instant值后手动转换:myObject.getCreationDate().atZone(ZoneOffset.UTC).toLocalDateTime()。
  • 此方法是全局性的,会影响所有Jackson对Instant类型的反序列化。

3. 创建自定义反序列化器

如果前两种方法不适用,例如你必须将时间戳反序列化为LocalDateTime而不是Instant,或者需要更复杂的转换逻辑,可以实现一个自定义的Jackson反序列化器。

实现步骤:

  1. 创建一个继承自StdDeserializer(或你目标类型)的类。
  2. 重写deserialize()方法,在该方法中手动解析JSON节点并执行转换逻辑。
  3. 使用@JsonDeserialize(using = YourDeserializer.class)注解将自定义反序列化器应用到目标字段上。

示例代码:

立即学习Java免费学习笔记(深入)”;

首先,定义自定义反序列化器:

import com.fasterxml.jackson.core.JacksonException;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.deser.std.StdDeserializer;
import java.io.IOException;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneOffset;

public class DateTimeDeserializer extends StdDeserializer {

    public DateTimeDeserializer() {
        super(LocalDateTime.class);
    }

    @Override
    public LocalDateTime deserialize(JsonParser p,
                                     DeserializationContext ctxt) throws IOException, JacksonException {
        JsonNode node = p.getCodec().readTree(p);
        long timestamp = node.longValue(); // 获取Epoch毫秒值

        return Instant
            .ofEpochMilli(timestamp)
            .atZone(ZoneOffset.UTC) // 同样,明确指定时区
            .toLocalDateTime();
    }
}

然后,在MyLocalApplicationClass中应用此反序列化器:

import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import java.time.LocalDateTime;

public class MyLocalApplicationClass {

    private String name;

    @JsonDeserialize(using = DateTimeDeserializer.class) // 应用自定义反序列化器
    @JsonProperty("creation_date")
    private LocalDateTime creationDate;

    @JsonProperty("created_by")
    private String createdBy;

    // Getters and Setters (省略)
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
    public LocalDateTime getCreationDate() { return creationDate; }
    public void setCreationDate(LocalDateTime creationDate) { this.creationDate = creationDate; }
    public String getCreatedBy() { return createdBy; }
    public void setCreatedBy(String createdBy) { this.createdBy = createdBy; }
}

注意事项:

  • 自定义反序列化器提供了最大的灵活性,可以处理任何复杂的转换逻辑。
  • 它只作用于被@JsonDeserialize注解标记的特定字段,不会影响其他字段或类。
  • 需要编写更多的代码来创建和维护反序列化器类。

总结

将REST响应中的Epoch毫秒时间戳反序列化为java.time类型是常见的需求。根据你的具体场景和偏好,可以选择以下策略:

  • 构造函数解析: 适用于少量、特定类的局部控制,代码相对直观,但可能导致重复。
  • 全局配置(使用Instant): 对于Spring Boot应用,这是最推荐和简洁的方式。它利用了Jackson的强大功能,通过配置将Epoch毫秒直接映射到Instant类型,减少了样板代码。之后可以根据需要将Instant转换为LocalDateTime或LocalDate。
  • 自定义反序列化器: 提供最大的灵活性和精确控制,适用于需要将Epoch毫秒直接映射到LocalDateTime或LocalDate,或存在复杂转换逻辑的场景。

无论选择哪种方法,关键在于理解java.time类型与Epoch毫秒之间的转换关系,特别是时区(ZoneOffset.UTC是常用选择)在转换过程中的作用,以确保日期时间数据的准确性。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

841

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

742

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

738

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

0

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.7万人学习

C# 教程
C# 教程

共94课时 | 7.2万人学习

Java 教程
Java 教程

共578课时 | 48.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号