锋盈数科-知识库 Logo
首页
软件开发
计算机基础
Hello Halo
新手必读
关于本知识库
登录 →
锋盈数科-知识库 Logo
首页 软件开发 计算机基础 Hello Halo 新手必读 关于本知识库
登录
  1. 首页
  2. 软件开发
  3. JAVA
  4. Spring Boot统一处理前后台传输时间格式统一问题(时间戳)

Spring Boot统一处理前后台传输时间格式统一问题(时间戳)

0
  • JAVA
  • 发布于 2024-08-15
  • 0 次阅读
黄健
黄健

本文由 简悦 SimpRead 转码, 原文地址 blog.csdn.net

  1. 使用 LocalDateDeserializer 与 LocalDateTimeDeserializer 对前端传输的时间戳格式进行统一反序列成 LocalDate 或者 LocalDateTime 类型,这样有利于前端在传输时间时统一者传时间戳,后端在接收时统一使用 LocalDate 或者 LocalDateTime 类型来接收,这样就不会因为格式的问题而烦脑。
/**
 * 扩展jackson反序列, 使其支持时间戳转LocalDate
 *
 * @author: mxj
 */
public class LocalDateExtDeserializer extends LocalDateDeserializer {
    public LocalDateExtDeserializer() {
        super(DateTimeFormatter.ISO_LOCAL_DATE);
    }
 
    public LocalDateExtDeserializer(DateTimeFormatter dtf) {
        super(dtf);
    }
 
    @Override
    protected JsonDeserializer<LocalDate> withDateFormat(DateTimeFormatter formatter) {
        return new LocalDateExtDeserializer(formatter);
    }
 
    @Override
    public LocalDate deserialize(JsonParser parse, DeserializationContext context) throws IOException {
        if (parse.hasToken(JsonToken.VALUE_NUMBER_INT)) {
            return LocalDateTime.ofInstant(Instant.ofEpochMilli(parse.getLongValue()), ZoneId.systemDefault()).toLocalDate();
        }
        return super.deserialize(parse, context);
    }
}
/**
 * 扩展jackson反序列, 使其支持时间戳转LocalDateTime
 *
 * @author: mxj
 */
public class LocalDateTimeExtDeserializer extends LocalDateTimeDeserializer {
    public LocalDateTimeExtDeserializer() {
        super(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
    }
 
    public LocalDateTimeExtDeserializer(DateTimeFormatter formatter) {
        super(formatter);
    }
 
    @Override
    protected JsonDeserializer<LocalDateTime> withDateFormat(DateTimeFormatter formatter) {
        return new LocalDateTimeExtDeserializer(formatter);
    }
 
    @Override
    public LocalDateTime deserialize(JsonParser parse, DeserializationContext context) throws IOException {
        if (parse.hasToken(JsonToken.VALUE_NUMBER_INT)) {
            return LocalDateTime.ofInstant(Instant.ofEpochMilli(parse.getLongValue()), ZoneId.systemDefault());
        }
        return super.deserialize(parse, context);
    }
 
}
  1. 使用 LocalDateSerializerh 或者 LocalDateTimeSerializer 对后端接口中 LocalDate 或者 LocalDateTime 类型返回给前端时间进行统一序列化成时间戳,这样对于前端来说可以根据时间戳统一进行封装进行统一处理。
package com.cell.jackson.serializer;
 
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer;
 
import java.io.IOException;
import java.time.LocalDate;
import java.time.ZoneOffset;
import java.time.format.DateTimeFormatter;
 
/**
 * 将LocalDate转为时间戳
 *
 * @author: mxj
 */
public class LocalDateExtSerializer extends LocalDateSerializer {
    private final Boolean writeDatesAsTimestamps;
 
    public LocalDateExtSerializer(Boolean writeDatesAsTimestamps) {
        super(DateTimeFormatter.ISO_LOCAL_DATE);
        this.writeDatesAsTimestamps = writeDatesAsTimestamps;
    }
 
    protected LocalDateExtSerializer(LocalDateSerializer base,
                                     Boolean useTimestamp, DateTimeFormatter dtf, JsonFormat.Shape shape, Boolean writeDatesAsTimestamps) {
        super(base, useTimestamp, dtf, shape);
        this.writeDatesAsTimestamps = writeDatesAsTimestamps;
    }
 
    @Override
    protected LocalDateSerializer withFormat(Boolean useTimestamp, DateTimeFormatter dtf, JsonFormat.Shape shape) {
        return new LocalDateExtSerializer(this, useTimestamp, dtf, shape, writeDatesAsTimestamps);
    }
 
    @Override
    public void serialize(LocalDate localDate, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException {
        if (writeDatesAsTimestamps) {
            jsonGenerator.writeNumber(localDate.atStartOfDay(ZoneOffset.systemDefault()).toInstant().toEpochMilli());
        } else {
            super.serialize(localDate, jsonGenerator, serializerProvider);
        }
    }
}
package com.cell.jackson.serializer;
 
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
 
import java.io.IOException;
import java.time.LocalDateTime;
import java.time.OffsetDateTime;
import java.time.format.DateTimeFormatter;
 
/**
 * 将LocalDateTime转为时间戳
 *
 * @author: mxj
 */
public class LocalDateTimeExtSerializer extends LocalDateTimeSerializer {
    private final Boolean writeDatesAsTimestamps;
 
    public LocalDateTimeExtSerializer(Boolean writeDatesAsTimestamps) {
        super(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
        this.writeDatesAsTimestamps = writeDatesAsTimestamps;
    }
 
    protected LocalDateTimeExtSerializer(DateTimeFormatter f, Boolean writeDatesAsTimestamps) {
        super(f);
        this.writeDatesAsTimestamps = writeDatesAsTimestamps;
    }
 
    @Override
    protected LocalDateTimeSerializer withFormat(Boolean useTimestamp, DateTimeFormatter f, JsonFormat.Shape shape) {
        return new LocalDateTimeExtSerializer(f, writeDatesAsTimestamps);
    }
 
    @Override
    public void serialize(LocalDateTime localDateTime, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException {
        if (writeDatesAsTimestamps) {
            jsonGenerator.writeNumber(localDateTime.toInstant(OffsetDateTime.now().getOffset()).toEpochMilli());
        } else {
            super.serialize(localDateTime, jsonGenerator, serializerProvider);
        }
    }
}
  1. 配置好以上两步后, 要在 WebMvcConfigurer 配置类中进行增加 RequestBody 内容转换器(缺点就是只针对于 Controller 层中的接中 @RequestBody 注解中的参数生效,解决方案在后头)
/**
 * WebMvcConfig
 *
 * @author: mxj
 * @Date 2023/1/28 11:29
 */
@ConditionalOnClass(WebMvcConfigurer.class)
public class WebMvcConfiguration implements WebMvcConfigurer {
 
    /**
     * 增加了Jackson对Jdk8Time的支持
     *
     * @param converters RequestBody内容转换器列表, 这里只需要对jackson提供的转换器进行拦截
     */
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        for (HttpMessageConverter<?> converter : converters) {
            if (MappingJackson2HttpMessageConverter.class.isAssignableFrom(converter.getClass())) {
                ObjectMapper objectMapper = ((MappingJackson2HttpMessageConverter) converter).getObjectMapper();
                SimpleModule dateModule = new SimpleModule();
                //jackson序列化
                dateModule.addSerializer(LocalTime.class, new LocalTimeSerializer(DateTimeFormatter.ofPattern("HH:mm:ss")));
                dateModule.addSerializer(LocalDate.class, new LocalDateExtSerializer(true));
                dateModule.addSerializer(LocalDateTime.class, new LocalDateTimeExtSerializer(true));
                //jackson反序列化, 默认的格式为 yyyy-MM-ddTHH:mm:ss, 故需要配置
                dateModule.addDeserializer(LocalDate.class, new LocalDateExtDeserializer());
                dateModule.addDeserializer(LocalDateTime.class, new LocalDateTimeExtDeserializer());
                objectMapper.registerModule(dateModule);
            }
        }
    }
}
  1. 在第 3 点中的缺点,就是要解决掉 @PathVariable 与 @RequestParam 等注解的接收参数时间转换问题(以 LongLocalDateTime 为例子)。利以 Converter 将前端数据转换为 java 对应的对象
package com.cell.converter;
 
import com.cell.converter.handler.localDateTime.LocalDateTimeConverterHandler;
import com.cell.converter.handler.localDateTime.LongLocalDateTimeConverter;
import com.cell.converter.handler.localDateTime.StringLocalDateTimeConverter;
import com.cell.webExport.config.response.exception.BusinessException;
import lombok.extern.slf4j.Slf4j;
import org.springframework.core.convert.converter.Converter;
 
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.List;
 
/**
 * 统一转换成LocalDateTime
 *
 * @Author MXJ
 */
@Slf4j
public class LocalDateTimeConverter implements Converter<String, LocalDateTime> {
    private static List<LocalDateTimeConverterHandler> localDateTimeConverterHandlerList = new ArrayList<>();
 
    static {
        //可以配置多种类型格式解释(这理配置了两种格式)
        localDateTimeConverterHandlerList.add(new StringLocalDateTimeConverter());
        localDateTimeConverterHandlerList.add(new LongLocalDateTimeConverter());
    }
 
 
    @Override
    public LocalDateTime convert(String source) {
 
        for (LocalDateTimeConverterHandler item : localDateTimeConverterHandlerList) {
            try {
                return item.getLocalDateTime(source);
            } catch (Exception e) {
                log.info("LocalDateTime时间格式解释失败");
            }
        }
        throw new BusinessException("LocalDateTime时间格式解释失败");
    }
}
package com.cell.converter.handler.localDateTime;
 
import java.time.LocalDateTime;
 
/**
 * 解释类型接口
 *
 * @Author MXJ
 */
public interface LocalDateTimeConverterHandler {
    /**
     * 获取LocalDateTime
     */
    LocalDateTime getLocalDateTime(String source);
}
package com.cell.converter.handler.localDateTime;
 
import lombok.extern.slf4j.Slf4j;
 
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneId;
 
/**
 * LocalDateTime时间戳类型(根据时间戳解释成LocalDateTime)
 *
 * @Author MXJ
 */
@Slf4j
public class LongLocalDateTimeConverter implements LocalDateTimeConverterHandler {
 
    /**
     * 获取LocalDateTime
     */
    @Override
    public LocalDateTime getLocalDateTime(String source) {
 
        Instant instant = Instant.ofEpochMilli(Long.parseLong(source));
        return LocalDateTime.ofInstant(instant, ZoneId.systemDefault());
 
    }
}
package com.cell.converter.handler.localDateTime;
 
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
 
/**
 * LocalDateTime字符串类型
 * @Author MXJ
 */
public class StringLocalDateTimeConverter implements LocalDateTimeConverterHandler{
 
    /**
     * 获取LocalDateTime(不同种格式)
     * @param source
     * @return
     */
    @Override
    public LocalDateTime getLocalDateTime(String source) {
        try {
            DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yy-M-d ah:mm");
            return LocalDateTime.parse(source, formatter);
        }catch (Exception e){
            DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
            return LocalDateTime.parse(source, formatter);
        }
    }
}
  1. 以上就是统一实现前后端时间格式问题(请大家多多指教)
标签: #软件开发 1171 #JAVA 991
相关文章

Spring 实现 3 种异步接口 2024-10-18 09:07

大家好,我是苏三~ 如何处理比较耗时的接口? 这题我熟,直接上异步接口,使用 Callable、WebAsyncTask 和 DeferredResult、CompletableFuture等均可实现。 但这些方法有局限性,处理结果仅返回单个值。在某些场景下,如果需要接口异步处理的同时,还持续不断地

重学SpringBoot3-集成Redis(五)之布隆过滤器 2024-10-08 11:24

更多SpringBoot3内容请关注我的专栏:《SpringBoot3》 期待您的点赞👍收藏⭐评论✍ 重学SpringBoot3-集成Redis(五)之布隆过滤器 1. 什么是布隆过滤器? * 基本概念 适用场景 2. 使用 Redis 实现布隆过滤器 * 项目依赖 Redis 配置

SpringBoot整合异步任务执行 2024-10-08 11:24

同步任务: 同步任务是在单线程中按顺序执行,每次只有一个任务在执行,不会引发线程安全和数据一致性等 并发问题 同步任务需要等待任务执行完成后才能执行下一个任务,无法同时处理多个任务,响应慢,影响用 户体验 异步任务: 异步任务是在多线程中同时执行,多个任务可以并发执行,同时处理多个请求,响应快,资源

springboot kafka多数据源,通过配置动态加载发送者和消费者 2024-10-08 11:24

前言 最近做项目,需要支持kafka多数据源,实际上我们也可以通过代码固定写死多套kafka集群逻辑,但是如果需要不修改代码扩展呢,因为kafka本身不处理额外逻辑,只是起到削峰,和数据的传递,那么就需要对架构做一定的设计了。 准备test kafka本身非常容易上手,如果我们需要单元测试,引入ja

SpringBoot 集成 Redis 2024-10-08 11:24

一:SpringBoot 集成 Redis ①Redis是一个 NoSQL(not only)数据库, 常作用缓存 Cache 使用。 ②Redis是一个中间件、是一个独立的服务器;常用的数据类型: string , hash ,set ,zset , list ③通过Redis客户端可以使用多种语

SpringBoot整合QQ邮箱 2024-10-08 11:24

SpringBoot可以通过导入依赖的方式集成多种技术,这当然少不了我们常用的邮箱,现在本章演示SpringBoot整合QQ邮箱发送邮件…. 下面按步骤进行: 1.获取QQ邮箱授权码 1.1 登录QQ邮箱 1.2 开启SMTP服务 找到下图中的SMTP服务区域,如果当前账号未开启的话自己手动开启。

目录

IT 外包服务商

  • 意见投递
  • zyf6619

软件开发应用

主菜单

  • 首页
  • 软件开发
  • 计算机基础
  • Hello Halo
  • 新手必读
  • 关于本知识库
Copyright © 2024 your company All Rights Reserved. Powered by Halo.