Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 312|回复: 0

彻底解决Spring mvc中时间的转换和序列化等问题

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-6-5 19:16:50 | 显示全部楼层 |阅读模式

    痛点

    在使用Spring mvc 进行开发时我们经常遇到前端传来的某种格式的时间字符串无法用java8的新特性java.time包下的具体类型参数来直接接收。 我们使用含有java.time封装类型的参数接收也会报反序列化问题,在返回前端带时间类型的同样会出现一些格式化的问题。今天我们来彻底解决他们。

    建议

    其实最科学的建议统一使用时间戳来代表时间。这个是最完美的,避免了前端浏览器的兼容性问题,同时也避免了其它一些中间件的序列化/反序列化问题。但是用时间表达可能更清晰语义化。两种方式各有千秋,如果我们坚持使用java8的时间类库也不是没有办法。下面我们会以java.time.LocalDateTime为例逐一解决这些问题。

    局部注解方式

    网上有很多文章说该注解是前端指向后端的,也就是前端向后端传递时间参数格式化使用的,这没有错!但是有一个小问题,该方式只能适用于不涉及反序列化的情况下。也就是以下场景才适用:

        @GetMapping("/local")
        public Map<String, String> data(@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss") LocalDateTime localDateTime) {
            Map<String, String> map = new HashMap<>(1);
            map.put("data", localDateTime.format(DateTimeFormatter.ofPattern("yyyy-MM-dd")));
            return map;
        }
    
    

    如果你在下面这个场景使用就不行了:

    
    @Data
    public class UserInfo {
    
        @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
        private LocalDateTime birthday;
        private String name;
        private Integer age;
    }
    
    
       @PostMapping("/user")
        public Object postData(@RequestBody UserInfo userInfo) {
            System.out.println("userInfo = " + userInfo);
            return userInfo;
        }
    

    原因是Post请求参数在body中,需要反序列化成对象。默认是jackson类库来进行反序列化,并不触发@DateTimeFormat注解机制。 这时我们就需要使用jackson的格式化注解@JsonFormat。我们将实体类UserInfo改造成下面的就可以了:

    @Data
    public class UserInfo {
    
        @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
        @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
        private LocalDateTime birthday;
        private String name;
        private Integer age;
    }
    

    以上两个注解可以并存,但是一定要清楚各自的使用场景。这里还有一个小细节:格式一定要对应好时间类型。比如yyyy-MM-dd 对应java.time.LocalDate 。如果再个性化一些@JsonFormat 可以被@JsonDeserialize@JsonSerialize 代替。但是它们的using参数需要你自己实现为你对应的时间类型类型。 如果@JsonFormat@JsonDeserialize@JsonSerialize同时存在@JsonFormat的优先级要更高。

    局部处理的好处

    局部处理的好处在于八个字:百花齐放,百家争鸣 。可以保持多样性、个性化 。但是局部带来了一个新的问题 :没有共同的标准 、不兼容。进而不方便维护。所以有时候基于业务需要我们全局化可以统一管理。下面我们将讲解如何进行全局化配置。

    全局化化时间格式配置

    全局化其实也是基于 @DateTimeFormat@JsonFormat 两种场景来进行配置。对于@DateTimeFormat的场景我们通过实现Spring提供的接口:

    DateTimeFormatter :

         // 时间格式化
        private static final DateTimeFormatter FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss", Locale.CHINA);
    
    

    类型转换接口:

     org.springframework.core.convert.converter.Converter<S,T>
    

    实现:

        @Bean
        public Converter<String, LocalDateTime> localDateConverter() {
            return new Converter<String, LocalDateTime>() {
                @Override
                public LocalDateTime convert(String source) {
                    return LocalDateTime.parse(source, FORMATTER);
                }
            };
        }
    

    或者格式化接口:

     org.springframework.format.Formatter<T>
    

    实现 :

        @Bean
        public Formatter<LocalDateTime> localDateFormatter() {
            return new Formatter<LocalDateTime>() {
                @Override
                public LocalDateTime parse(String text, Locale locale) throws ParseException {
                    return LocalDateTime.parse(text, FORMATTER);
                }
    
                @Override
                public String print(LocalDateTime object, Locale locale) {
                    return object.format(FORMATTER);
                }
            };
        }
    

    以上两个接口的实现都要注册为Spring Bean,配置的时候二者选其一即可,其中S即Source也就是来源,其实就是前端的时间字符串。T即Target也就是目标,代表你需要转化或者格式化的时间java类型。

    那么对于时间序列化和反序列化我们进行如下配置就行了(基于默认jackson,以LocalDateTime 为例):

        @Bean
        public Jackson2ObjectMapperBuilderCustomizer jackson2ObjectMapperBuilderCustomizer() {
    
            return jacksonObjectMapperBuilder -> jacksonObjectMapperBuilder
                     // 反序列化
                    .deserializerByType(LocalDateTime.class, new LocalDateTimeDeserializer(FORMATTER))
                     // 序列化
                    .serializerByType(LocalDateTime.class, new LocalDateTimeSerializer(FORMATTER));
        }
    
    

    同样该jsonMapper自定义构建器要注册成Spring Bean才行。

    全局配置要点

    全局配置的一些优缺点上面已经阐述了,这里我还是要啰嗦一下要点避免你踩坑。全局配置跟局部配置一样。同样要约定pattern。这就要求我们全局保持一致。我们可以实现多个以上的全局配置来对其他诸如LocalDateOffsetDateTime 的适配。同时如果我们接入了其它一些需要用到序列化/反序列化的中间件,比如redis、rabbitmq,我们也要注意进行适配。

    总结

    通过以上对时间格式的局部和全局处理方式的介绍,相信困扰你的Spring mvc 时间问题不会再存在了。如果感觉写对可以请转发告诉其他同学,点个赞,关注一下。

    关注公众号:码农小胖哥 获取更多资讯

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-2-2 13:03 , Processed in 0.105663 second(s), 27 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表