0

0

SpringBoot JSON全局日期格式转换器如何实现

PHPz

PHPz

发布时间:2023-05-23 08:07:18

|

1599人浏览过

|

来源于亿速云

转载

    需求

    前台有日期字符串的数据,提交到后台。后台实体类使用date属性接收。
    日期字符串有多种格式,需要用一个转换器将合法的日期字符串格式转换为date类型。

    分析

    当前台的提交数据的Content-Typeapplication/json;charset=utf-8,后台使用@RequestBody来接收数据的时候,使用此转换方式。

    一. 前期准备

    1.1 日期正则注解

    用来标记日期字符串所对应的正则表达式

    import java.lang.annotation.*;
    
    @Documented
    @Retention(RetentionPolicy.RUNTIME)
    @Target({ElementType.FIELD})
    public @interface DatePattern {
    	
        String value();
    }

    1.2 日期格式定数

    指定所有支持的日期格式

    public final class DateFormatPart {
    
    	@DatePattern("^\\d{4}$")
    	public static final String YYYY = "yyyy";
    
    	@DatePattern("^\\d{4}\\d{2}$")
    	public static final String YYYYMM = "yyyyMM";
    
    	@DatePattern("^\\d{4}/\\d{2}$")
    	public static final String YYYYMM_SLASH = "yyyy/MM";
    
    	@DatePattern("^\\d{4}\\d{1,2}\\d{1,2}$")
    	public static final String YYYYMMDD = "yyyyMMdd";
    
    	@DatePattern("^\\d{4}/\\d{2}/\\d{2}$")
    	public static final String YYYYMMDD_SLASH = "yyyy/MM/dd";
    
    	@DatePattern("[0-9]+\\u5e74[0-9]+\\u6708[0-9]+\\u65e5$")
    	public static final String YYYYMMDD_JP = "yyyy年MM月dd日";
    
        @DatePattern("^\\d{4}\\d{1,2}\\d{1,2} \\d{1,2}:\\d{1,2}:\\d{1,2}$")
    	public static final String YYYYMMDD_HHMMSS = "yyyyMMdd HH:mm:ss";
    
        @DatePattern("^\\d{4}/\\d{2}/\\d{2} \\d{1,2}:\\d{1,2}:\\d{1,2}$")
    	public static final String YYYYMMDD_HHMMSS_SLASH = "yyyy/MM/dd HH:mm:ss";
    }

    1.3 日期转换工具类

    • 从日期格式定数类中获取所有的属性值和该属性上所标记的正则注解,通过反射来映射为map。

    • 如果有需要增删的日期格式的话,只需要修改日期格式定数即可,便于维护。

    import org.springframework.util.ObjectUtils;
    import org.springframework.util.ReflectionUtils;
    
    import java.lang.reflect.Field;
    import java.text.ParseException;
    import java.text.SimpleDateFormat;
    import java.util.*;
    
    public final class DateUtil {
    
        // 日期格式 <==> 正则的map,使用LinkedHashMap可以避免按照顺序匹配正则表达式
        private static final Map datePatternMap = new LinkedHashMap<>();
    
        // 日期格式List
        private static final List dateFormatList = new ArrayList<>();
    
        // 使用静态代码块,可以保证只初始化一次
        static {
    
            Class dateFormatClass = DateFormatPart.class;
            // 获取所有的属性
            Field[] fields = dateFormatClass.getFields();
            for (Field field : fields) {
    
                // 获取属性上的注解
                DatePattern annotation = field.getAnnotation(DatePattern.class);
                if (ObjectUtils.isEmpty(annotation)) {
                    continue;
                }
    
                //  强制让属性可以访问
                ReflectionUtils.makeAccessible(field);
    
                // 日期格式化字符串
                String dateFormatStr = "";
                try {
                    // 获取当前属性所对应的属性值
                    dateFormatStr = (String)field.get(dateFormatClass);
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                }
    
                dateFormatList.add(dateFormatStr);
    
                // 将该属性的属性值和属性上的正则表达式放到map中
                datePatternMap.put(dateFormatStr, annotation.value());
            }
        }
    	
    	// 用所有可能支持的格式将日期字符串转换为Date格式
        public static Date formatDateStrToDateAllFormat(String dateStr) {
    
            if (ObjectUtils.isEmpty(dateStr)) {
                return null;
            }
    
            try {
                for (Map.Entry mapEntry : datePatternMap.entrySet()) {
    				
    				// 如果当前日期字符串不符合当前正则的话
                    if (!dateStr.matches(mapEntry.getValue())) {
                        continue;
                    }
    
                    return DateUtil.formatStringToDate(dateStr, mapEntry.getKey());
                }
            } catch (ParseException e) {
                return null;
            }
    
            return null;
        }
    	
    	// 通过指定的格式将日期字符串转换为Date类型
        public static Date formatStringToDate(String dateStr, String format) throws ParseException {
    
            if (ObjectUtils.isEmpty(dateStr) || !dateFormatList.contains(format)) {
                return null;
            }
    
            SimpleDateFormat time = new SimpleDateFormat(format);
            return time.parse(dateStr);
        }
    }

    二. 方式1-继承DateDeserializer类,重写_parseDate方法

    • 该方式的要点是通过继承DateDeserializer类,然后重写_parseDate方法实现转换功能

    • 自定义的GlobalDateDeserializer类需要添加到自定义的SimpleModule模块中,然后将模块添加到ObjectMapper中,通过jackson来实现转换。

    • 通过设置ObjectMapper对象的setDateFormat方法来实现后台数据返回到前台时的Date转String的默认格式。

    import com.fasterxml.jackson.core.JsonParser;
    import com.fasterxml.jackson.core.json.PackageVersion;
    import com.fasterxml.jackson.databind.DeserializationContext;
    import com.fasterxml.jackson.databind.DeserializationFeature;
    import com.fasterxml.jackson.databind.ObjectMapper;
    import com.fasterxml.jackson.databind.SerializationFeature;
    import com.fasterxml.jackson.databind.deser.std.DateDeserializers.DateDeserializer;
    import com.fasterxml.jackson.databind.module.SimpleModule;
    import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.context.annotation.Primary;
    import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
    
    import java.io.IOException;
    import java.text.DateFormat;
    import java.text.SimpleDateFormat;
    import java.util.Date;
    
    @Configuration
    public class CustomConfig {
    
        /*
         * 自定义的全局的日期转换解析类,需要继承jackson包中的DateDeserializer
         * 因为此类不会在别的地方使用了,因此直接使用内部类聚合到自定义的配置文件中
         */
        private static final class GlobalDateDeserializer extends DateDeserializer {
    
            @Override
            protected Date _parseDate(JsonParser jp, DeserializationContext context) throws IOException {
    
                return DateUtil.formatDateStrToDateAllFormat(jp.getText());
            }
        }
    
        @Bean("objectMapper")
        @Primary
        @ConditionalOnMissingBean
        public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
    
            // 创建jackson对象
            ObjectMapper jackson = builder.createXmlMapper(false).build();
    
            /*
             * DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES
             * 在进行序列化或者反序列化的时候,
             * JSON字符串中有一个字段,但是我们的对象没有这个字段的时候,该怎么处理
             * 设置为true的时候,会抛出异常
             * 设置为false,忽略异常继续处理
             */
            jackson.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
            // 禁用默认的「yyyy-MM-dd'T'HH:mm:ss.SSS」UTC类型
            jackson.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
    
             /*
             * 设置序列化时的默认格式,即后台返回数据到前台的时候,
             * Date类型数据需要转换为的字符串格式
             * 如果不进行指定的话,默认使用 yyyy-MM-dd'T'HH:mm:ss.SSS 格式
             */
            DateFormat dateFormat = new SimpleDateFormat(DateFormatPart.YYYYMMDD_HHMMSS_SLASH);
            jackson.setDateFormat(dateFormat);
    
            // 创建一个模块,指定该模块是用来解析 Date.class 类型数据的
            SimpleModule newModule = new SimpleModule("GlobalDateDeserializer", PackageVersion.VERSION);
            // 将我们创建的全局日期转换类添加到模块中,指定转换Date类型
            newModule.addDeserializer(Date.class, new CustomConfig.GlobalDateDeserializer());
            // 将该模块添加到jackson中
            jackson.registerModule(newModule);
    
            return jackson;
        }
    }

    三. 方式2-继承StdDateFormat类,重写方法

    • parse方法用来将日期字符串转换为Date(前台向后台传数据)

      燕雀Logo
      燕雀Logo

      为用户提供LOGO免费设计在线生成服务

      下载
    • format方法用来将Date格式的数据转换为指定格式的字符串(后台向前台传数据)。

    import java.text.FieldPosition;
    import java.text.ParsePosition;
    import java.text.SimpleDateFormat;
    import java.util.Date;
    
    import com.fasterxml.jackson.databind.util.StdDateFormat;
    
    public class GlobalJsonDateConvert extends StdDateFormat {
    
        // 静态初始化final,共享
        public static final GlobalJsonDateConvert instance = new GlobalJsonDateConvert();
    
        // 日期字符串解析为日期
        @Override
        public Date parse(String dateStr, ParsePosition pos) {
            return getDate(dateStr);
        }
    
        @Override
        public Date parse(String dateStr) {
            return getDate(dateStr);
        }
    	
    	// 使用自定义的日期转换工具类将所有可能支持的日期字符串转换为Date格式
        private Date getDate(String dateStr) {
            return DateUtil.formatDateStrToDateAllFormat(dateStr);
        }
        
    	/*
       	 * 设置序列化时的默认格式,即后台返回数据到前台的时候,
         * Date类型数据需要转换为的字符串格式
         * 如果不进行指定的话,默认使用 yyyy-MM-dd'T'HH:mm:ss.SSS 格式
         */
        @Override
        public StringBuffer format(Date date, StringBuffer toAppendTo, FieldPosition fieldPosition){
            SimpleDateFormat sdf = new SimpleDateFormat(DateFormatPart.YYYYMMDD_HHMMSS_SLASH);
            return sdf.format(date, toAppendTo, fieldPosition);
        }
    
        @Override
        public GlobalJsonDateConvert clone() {
            super.clone();
            return new GlobalJsonDateConvert();
        }
    }

    3.1 MappingJackson2HttpMessageConverter方式

    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;
    
    @Configuration
    public class CustomConfig {
    
        // JSON格式 全局日期转换器配置
        @Bean
        public MappingJackson2HttpMessageConverter createMappingJackson2HttpMessageConverter() {
    
            /*
             * 通过MappingJackson2HttpMessageConverter得到的ObjectMapper,
             * 已经默认把 DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES 给关闭了
             */
            MappingJackson2HttpMessageConverter jacksonConverter = new MappingJackson2HttpMessageConverter();
            jacksonConverter.getObjectMapper().setDateFormat(GlobalJsonDateConvert.instance);
    
            return jacksonConverter;
        }
    }

    3.2 ObjectMapper方式

    通过Jackson2ObjectMapperBuilder创建ObjectMapper对象,然后将我们定义的转换器GlobalJsonDateConvert放到ObjectMapper对象中

    import com.fasterxml.jackson.databind.ObjectMapper;
    import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.context.annotation.Primary;
    import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
    
    @Configuration
    public class CustomConfig {
    
        @Bean("objectMapper")
        @Primary
        @ConditionalOnMissingBean
        public ObjectMapper objectMapper(Jackson2ObjectMapperBuilder builder) {
    		
            ObjectMapper objectMapper = builder.build();
            objectMapper.setDateFormat(GlobalJsonDateConvert.instance);
            return objectMapper;
        }
    }

    3.3 Jackson2ObjectMapperBuilder方式

    将我们定义的转换器GlobalJsonDateConvert放到Jackson2ObjectMapperBuilder对象中

    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
    
    @Configuration
    public class CustomConfig {
    
        @Bean
        public Jackson2ObjectMapperBuilder objectMapper() {
    
            Jackson2ObjectMapperBuilder jackson2ObjectMapperBuilder = new Jackson2ObjectMapperBuilder();
            jackson2ObjectMapperBuilder.dateFormat(GlobalJsonDateConvert.instance);
            return jackson2ObjectMapperBuilder;
        }
    }

    四. 效果

    ⏹前台JS

    // 向后台传输的json数据
    const jsonData = {
    	// ????待处理的日期字符串数据
        birthday: '2027年12月12日',
        nameAA: 'jiafeitian',
        hobby: '吃饭'
    };
    
    $.ajax({
        url: url,
        type: 'POST',
        // 对象转换为json字符串
        data: JSON.stringify(jsonData),
        contentType: "application/json;charset=utf-8",
        success: function (data, status, xhr) {
            console.log(data);
        }
    });

    ⏹后台Form

    import lombok.Data;
    import java.util.Date;
    
    @Data
    public class Test15Form {
    
        private String name;
    
        private String hobby;
    
        private String address;
    	
    	// 用来接收的Date类型的数据
        private Date birthday;
    }

    可以看到前台提交的日期字符串被转换为Date格式了

    SpringBoot JSON全局日期格式转换器如何实现

    热门AI工具

    更多
    DeepSeek
    DeepSeek

    幻方量化公司旗下的开源大模型平台

    豆包大模型
    豆包大模型

    字节跳动自主研发的一系列大型语言模型

    通义千问
    通义千问

    阿里巴巴推出的全能AI助手

    腾讯元宝
    腾讯元宝

    腾讯混元平台推出的AI助手

    文心一言
    文心一言

    文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

    讯飞写作
    讯飞写作

    基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

    即梦AI
    即梦AI

    一站式AI创作平台,免费AI图片和视频生成。

    ChatGPT
    ChatGPT

    最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

    相关专题

    更多
    json数据格式
    json数据格式

    JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

    418

    2023.08.07

    json是什么
    json是什么

    JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

    535

    2023.08.23

    jquery怎么操作json
    jquery怎么操作json

    操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

    311

    2023.10.13

    go语言处理json数据方法
    go语言处理json数据方法

    本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

    77

    2025.09.10

    js正则表达式
    js正则表达式

    php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

    513

    2023.06.20

    正则表达式不包含
    正则表达式不包含

    正则表达式,又称规则表达式,,是一种文本模式,包括普通字符和特殊字符,是计算机科学的一个概念。正则表达式使用单个字符串来描述、匹配一系列匹配某个句法规则的字符串,通常被用来检索、替换那些符合某个模式的文本。php中文网给大家带来了有关正则表达式的相关教程以及文章,希望对大家能有所帮助。

    251

    2023.07.05

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

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

    746

    2023.07.05

    java正则表达式匹配字符串
    java正则表达式匹配字符串

    在Java中,我们可以使用正则表达式来匹配字符串。本专题为大家带来java正则表达式匹配字符串的相关内容,帮助大家解决问题。

    215

    2023.08.11

    俄罗斯Yandex引擎入口
    俄罗斯Yandex引擎入口

    2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

    158

    2026.01.28

    热门下载

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

    精品课程

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

    共17课时 | 2.4万人学习

    Swoft2.x速学之http api篇课程
    Swoft2.x速学之http api篇课程

    共16课时 | 0.9万人学习

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

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