详细分析Java中的@JsonFormat注解和@DateTimeFormat注解

目录

  • 前言
  • 1. @JsonFormat注解
  • 2. @DateTimeFormat注解
  • 3. Demo
    • 3.1 无注解
    • 3.2 有注解
  • 4. 拓展

前言

下文中涉及MybatisPlus的逻辑删除的知识,可看我之前这篇文章:详细讲解MybatisPlus实现逻辑删除

对应的Navicat设置数据库最新时间可看我这篇文章:Navicat 设置时间默认值(当前最新时间)


为了使 @JsonFormat 生效,项目必须引入 Jackson 库的相关依赖:
(如果是springboot项目,可不用配置,本身spring-boot-start-web依赖已包含)

<!-- JSON工具类 -->
<dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.12.6</version>
</dependency>

摘要:

  1. 注解@JsonFormat主要是后端到前端的时间格式的转换

  2. 注解@DateTimeFormat主要是前端到后端的时间格式的转换

1. @JsonFormat注解

@JsonFormat 是 Jackson 库中的注解,用于在序列化和反序列化过程中控制日期和时间的格式。

该注解提供了一种自定义日期和时间格式的方式,以确保在 JSON 数据和 Java 对象之间正确地进行转换。

以下是 @JsonFormat 注解的一些主要概念和功能:

  • pattern(模式): 通过 pattern 属性,您可以指定日期和时间的格式。
    例如,如果要将日期格式设置为"yyyy-MM-dd",可以使用 @JsonFormat(pattern = "yyyy-MM-dd")

  • timezone(时区): 使用 timezone 属性可以指定日期和时间的时区。这对于确保正确地处理跨时区的日期数据很重要。

  • locale(区域设置): 通过 locale 属性,您可以指定用于格式化的区域设置,以便支持不同的语言和地区。

  • shape(形状): shape 属性定义了序列化后的日期表示形式。例如,您可以将日期表示为字符串或时间戳。

  • with(特定类型的格式): 使用 with 属性,您可以为不同的 Java 类型指定不同的格式。这对于处理不同类型的日期数据非常有用。

下面是一个简单的 Java 类示例,演示如何使用 @JsonFormat 注解:

import com.fasterxml.jackson.annotation.JsonFormat;import java.util.Date;public class MyObject {@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+00:00")private Date myDate;// 其他属性和方法public Date getMyDate() {return myDate;}public void setMyDate(Date myDate) {this.myDate = myDate;}
}

在这个例子中,myDate 属性使用了 @JsonFormat 注解,指定了日期的格式和时区。

当这个对象被序列化成 JSON 或者从 JSON 反序列化时,将使用指定的格式来处理日期数据。

2. @DateTimeFormat注解

@DateTimeFormat 是 Spring 框架中用于处理日期和时间格式的注解。

它通常与 @RequestMapping@RequestParam 等注解一起使用,以指定接收或发送日期时间参数时的格式。

以下是 @DateTimeFormat 注解的一些主要概念和功能:

  • pattern(模式): 通过 pattern 属性,您可以指定日期和时间的格式。
    @JsonFormat 不同,@DateTimeFormat 是专门为 Spring 框架设计的,用于在 Web 请求中处理日期参数。

  • iso(ISO标准格式): 使用 iso 属性可以指定使用 ISO 标准的日期时间格式。
    例如,@DateTimeFormat(iso = ISO.DATE) 表示日期部分采用标准日期格式。

  • style(样式): style 属性定义了预定义的日期和时间格式。
    有三种样式可用:DEFAULTSHORTMEDIUMLONGFULL。这些样式在不同的地区设置下有不同的显示效果。

  • lenient(宽松解析): lenient 属性用于指定是否宽松解析日期。
    如果设置为 true,则在解析日期时会尽量接受不严格符合格式的输入。

  • patternResolver(模式解析器): patternResolver 属性允许您指定自定义的模式解析器,以便更灵活地处理日期时间格式。

下面是一个简单的 Spring MVC 控制器示例,演示如何使用 @DateTimeFormat 注解:

import org.springframework.format.annotation.DateTimeFormat;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;import java.util.Date;@RestController
@RequestMapping("/date")
public class DateController {@RequestMapping("/processDate")public String processDate(@RequestParam("myDate") @DateTimeFormat(pattern = "yyyy-MM-dd") Date myDate) {// 处理日期逻辑return "Received date: " + myDate;}
}

processDate 方法接收一个名为 myDate 的参数,并使用 @DateTimeFormat 注解指定了日期的格式。

当请求中包含名为 myDate 的参数时,Spring 将自动将参数解析为 Date 类型,并应用指定的格式。

3. Demo

为了做好例子的前提,需要配置好数据库以及代码信息。

数据库相应的信息如下:

数据库类型为datetime或者timestamp,根据时间戳更新:
在这里插入图片描述

代码信息主要如下:(实体类)

@Data
@AllArgsConstructor
@NoArgsConstructor
@TableName("test_student")
public class student{@TableId(value = "id", type = IdType.AUTO)private int id;private String username;// 其他字段...private Date time;@TableLogicprivate Integer deleteFlag;}

service类:

public interface StudentService extends IService<student> {// 这里可以自定义一些业务方法
}

实现类:

@Service
public class StrudentServiceimpl extends ServiceImpl<StudentMapper, student> implements StudentService {// 这里可以实现自定义的业务方法
}

Mapper类:

@Mapper
public interface StudentMapper extends BaseMapper<student> {// 这里可以自定义一些查询方法
}

3.1 无注解

对应的实体类在Date中没有相关的注解:private Date time;

对应没有注解的时候,测试类输出结果如下:time=Thu Jan 11 21:02:06 CST 2024

3.2 有注解

这两者的注解一般联合使用

@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone="GMT+8")
private Date time;

一般系统都是前后交互

对此总结如下:

  • 注解@JsonFormat主要是后端到前端的时间格式的转换

  • 注解@DateTimeFormat主要是前端到后端的时间格式的转换

4. 拓展

常用的配置如下:

在这里插入图片描述

对应的时间配置可以使用该类进行拓展:

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by FernFlower decompiler)
//import java.text.ParseException;
import java.time.Duration;
import java.time.Instant;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.Period;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.time.temporal.Temporal;
import java.time.temporal.TemporalAccessor;
import java.time.temporal.TemporalAmount;
import java.time.temporal.TemporalQuery;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.TimeZone;
import org.springframework.util.Assert;public class DateUtil {public static final String PATTERN_DATETIME = "yyyy-MM-dd HH:mm:ss";public static final String PATTERN_DATETIME_MINI = "yyyyMMddHHmmss";public static final String PATTERN_DATE = "yyyy-MM-dd";public static final String PATTERN_TIME = "HH:mm:ss";public static final ConcurrentDateFormat DATETIME_FORMAT = ConcurrentDateFormat.of("yyyy-MM-dd HH:mm:ss");public static final ConcurrentDateFormat DATETIME_MINI_FORMAT = ConcurrentDateFormat.of("yyyyMMddHHmmss");public static final ConcurrentDateFormat DATE_FORMAT = ConcurrentDateFormat.of("yyyy-MM-dd");public static final ConcurrentDateFormat TIME_FORMAT = ConcurrentDateFormat.of("HH:mm:ss");public static final DateTimeFormatter DATETIME_FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");public static final DateTimeFormatter DATETIME_MINI_FORMATTER = DateTimeFormatter.ofPattern("yyyyMMddHHmmss");public static final DateTimeFormatter DATE_FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd");public static final DateTimeFormatter TIME_FORMATTER = DateTimeFormatter.ofPattern("HH:mm:ss");public DateUtil() {}public static Date now() {return new Date();}public static Date plusYears(Date date, int yearsToAdd) {return set(date, 1, yearsToAdd);}public static Date plusMonths(Date date, int monthsToAdd) {return set(date, 2, monthsToAdd);}public static Date plusWeeks(Date date, int weeksToAdd) {return plus(date, Period.ofWeeks(weeksToAdd));}public static Date plusDays(Date date, long daysToAdd) {return plus(date, Duration.ofDays(daysToAdd));}public static Date plusHours(Date date, long hoursToAdd) {return plus(date, Duration.ofHours(hoursToAdd));}public static Date plusMinutes(Date date, long minutesToAdd) {return plus(date, Duration.ofMinutes(minutesToAdd));}public static Date plusSeconds(Date date, long secondsToAdd) {return plus(date, Duration.ofSeconds(secondsToAdd));}public static Date plusMillis(Date date, long millisToAdd) {return plus(date, Duration.ofMillis(millisToAdd));}public static Date plusNanos(Date date, long nanosToAdd) {return plus(date, Duration.ofNanos(nanosToAdd));}public static Date plus(Date date, TemporalAmount amount) {Instant instant = date.toInstant();return Date.from(instant.plus(amount));}public static Date minusYears(Date date, int years) {return set(date, 1, -years);}public static Date minusMonths(Date date, int months) {return set(date, 2, -months);}public static Date minusWeeks(Date date, int weeks) {return minus(date, Period.ofWeeks(weeks));}public static Date minusDays(Date date, long days) {return minus(date, Duration.ofDays(days));}public static Date minusHours(Date date, long hours) {return minus(date, Duration.ofHours(hours));}public static Date minusMinutes(Date date, long minutes) {return minus(date, Duration.ofMinutes(minutes));}public static Date minusSeconds(Date date, long seconds) {return minus(date, Duration.ofSeconds(seconds));}public static Date minusMillis(Date date, long millis) {return minus(date, Duration.ofMillis(millis));}public static Date minusNanos(Date date, long nanos) {return minus(date, Duration.ofNanos(nanos));}public static Date minus(Date date, TemporalAmount amount) {Instant instant = date.toInstant();return Date.from(instant.minus(amount));}private static Date set(Date date, int calendarField, int amount) {Assert.notNull(date, "The date must not be null");Calendar c = Calendar.getInstance();c.setLenient(false);c.setTime(date);c.add(calendarField, amount);return c.getTime();}public static String formatDateTime(Date date) {return DATETIME_FORMAT.format(date);}public static String formatDateTimeMini(Date date) {return DATETIME_MINI_FORMAT.format(date);}public static String formatDate(Date date) {return DATE_FORMAT.format(date);}public static String formatTime(Date date) {return TIME_FORMAT.format(date);}public static String format(Date date, String pattern) {return ConcurrentDateFormat.of(pattern).format(date);}public static String formatDateTime(TemporalAccessor temporal) {return DATETIME_FORMATTER.format(temporal);}public static String formatDateTimeMini(TemporalAccessor temporal) {return DATETIME_MINI_FORMATTER.format(temporal);}public static String formatDate(TemporalAccessor temporal) {return DATE_FORMATTER.format(temporal);}public static String formatTime(TemporalAccessor temporal) {return TIME_FORMATTER.format(temporal);}public static String format(TemporalAccessor temporal, String pattern) {return DateTimeFormatter.ofPattern(pattern).format(temporal);}public static Date parse(String dateStr, String pattern) {ConcurrentDateFormat format = ConcurrentDateFormat.of(pattern);try {return format.parse(dateStr);} catch (ParseException var4) {throw Exceptions.unchecked(var4);}}public static Date parse(String dateStr, ConcurrentDateFormat format) {try {return format.parse(dateStr);} catch (ParseException var3) {throw Exceptions.unchecked(var3);}}public static <T> T parse(String dateStr, String pattern, TemporalQuery<T> query) {return DateTimeFormatter.ofPattern(pattern).parse(dateStr, query);}public static Instant toInstant(LocalDateTime dateTime) {return dateTime.atZone(ZoneId.systemDefault()).toInstant();}public static LocalDateTime toDateTime(Instant instant) {return LocalDateTime.ofInstant(instant, ZoneId.systemDefault());}public static Date toDate(LocalDateTime dateTime) {return Date.from(toInstant(dateTime));}public static Date toDate(final LocalDate localDate) {return Date.from(localDate.atStartOfDay(ZoneId.systemDefault()).toInstant());}public static Calendar toCalendar(final LocalDateTime localDateTime) {return GregorianCalendar.from(ZonedDateTime.of(localDateTime, ZoneId.systemDefault()));}public static long toMilliseconds(final LocalDateTime localDateTime) {return localDateTime.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli();}public static long toMilliseconds(LocalDate localDate) {return toMilliseconds(localDate.atStartOfDay());}public static LocalDateTime fromCalendar(final Calendar calendar) {TimeZone tz = calendar.getTimeZone();ZoneId zid = tz == null ? ZoneId.systemDefault() : tz.toZoneId();return LocalDateTime.ofInstant(calendar.toInstant(), zid);}public static LocalDateTime fromInstant(final Instant instant) {return LocalDateTime.ofInstant(instant, ZoneId.systemDefault());}public static LocalDateTime fromDate(final Date date) {return LocalDateTime.ofInstant(date.toInstant(), ZoneId.systemDefault());}public static LocalDateTime fromMilliseconds(final long milliseconds) {return LocalDateTime.ofInstant(Instant.ofEpochMilli(milliseconds), ZoneId.systemDefault());}public static Duration between(Temporal startInclusive, Temporal endExclusive) {return Duration.between(startInclusive, endExclusive);}public static Period between(LocalDate startDate, LocalDate endDate) {return Period.between(startDate, endDate);}public static Duration between(Date startDate, Date endDate) {return Duration.between(startDate.toInstant(), endDate.toInstant());}public static String secondToTime(Long second) {if (second != null && second != 0L) {long days = second / 86400L;second = second % 86400L;long hours = second / 3600L;second = second % 3600L;long minutes = second / 60L;second = second % 60L;return days > 0L ? StringUtil.format("{}天{}小时{}分{}秒", new Object[]{days, hours, minutes, second}) : StringUtil.format("{}小时{}分{}秒", new Object[]{hours, minutes, second});} else {return "";}}public static String today() {return format(new Date(), "yyyyMMdd");}public static String time() {return format(new Date(), "yyyyMMddHHmmss");}public static Integer hour() {return NumberUtil.toInt(format(new Date(), "HH"));}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/614822.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL、Oracle、PostgreSQL 数据库备份用的 Shell 脚本

文章目录 一、概要二、MySQL 数据库备份与还原异地备份脚本2.1 MySQL 数据库异地备份脚本2.2 MySQL 数据库的恢复 三、Oracle 数据库备份与还原脚本3.1 Oracle 数据库异地备份脚本3.2 Oracle 数据库的恢复 四、PostgreSQL 数据库异地备份与还原脚本4.1 PostgreSQL 数据库备份脚…

生活自来水厂污水处理设备需要哪些

生活自来水厂是确保我们日常用水质量安全的重要设施。在自来水的生产过程中&#xff0c;污水处理设备是不可或缺的环节。那么&#xff0c;生活自来水厂的污水处理设备都有哪些呢&#xff1f;本文将为您详细介绍。 首先&#xff0c;生活自来水厂的污水处理设备主要包括预处理设备…

第11章 GUI Page489~494 步骤三十 保存画板文件

为“保存”菜单项 MenuItemFileSave挂接事件响应函数&#xff1a; 实际运行时&#xff0c;现版TrySaveFile()函数有点儿傻&#xff0c;点击保存菜单&#xff0c;还会弹出对话框&#xff0c;问我们“要不要保存” 修改TrySaveFile()函数 函数声明修改为&#xff1a; 函数实现修…

药品不良反应智能监测系统,java药品不良反应(ADR)智能监测系统源码,基于SpringBoot+Vue+MySQL技术开发

java药品不良反应智能监测系统 系统概述&#xff1a; 基于AI技术和深度学习模型&#xff0c;系统构建ADR智能识别模型&#xff0c;实现病程记录分析&#xff0c;提取病程记录实体&#xff0c;并完成语义关系的分类识别。一方面可以根据医务人员设置的监测任务或主题&#xff0c…

Open CV 图像处理基础:(六)在Java中使用 Open CV进行图片翻转和图片旋转

在Java中使用 Open CV进行图片翻转和图片旋转 目录 在Java中使用 Open CV进行图片翻转和图片旋转前言图片翻转函数代码示例其它翻转方向垂直翻转两轴翻转 图片旋转函数代码示例 Open CV 专栏导航 前言 在Java中使用OpenCV进行图片翻转和旋转是一种基本的图像处理技术&#xff0…

管理篇 - 24到28

学习目标&#xff1a; 为了更轻松的工作 学习内容&#xff1a; 新手管理最常犯的错&#xff1a;太把自己当回事儿&#xff01;当管理比自己干活还累&#xff1f;升职加薪后他真香了&#xff01;管理误区&#xff1a;要不要和下属做朋友&#xff1f;很多人都只理解了第一层&…

MySQL面试题 | 02.精选MySQL面试题

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

vim基本操作命令

一、vi简介 vi是“Visual interface”的简称&#xff0c;它在Linux上的地位就仿佛Edit程序在DOS上一样。它可以执行输出、删除、查找、替换、块操作等众多文本操作&#xff0c;而且用户可以根据自己的需要对其进行定制。Vi不是一个排版程序&#xff0c;它不象Word或WPS那样可以…

商城小程序(8.购物车页面)

目录 一、商品列表区域1、渲染购物车商品列表的标题区域2、渲染商品列表区域的基本结构3、为my-goods组件封装radio勾选状态4、为my-goods组件封装radio-change事件5、修改购物车中商品的选择状态6、为my-goods组件封装NumberBox7、为my-goods封装num-change事件8、修改购物车商…

解决opencv出错:ImportError: numpy.core.multiarray failed to import

利用安装好opencv后&#xff0c;导入cv2的时候报错&#xff0c;如下 from .cv2 import * ImportError: numpy.core.multiarray failed to import这里是numpy库的版本过低或者过高&#xff0c;不合适刚刚装的opencv版本。 尝试升高或降低numpy版本&#xff0c;一般是升级到高版…

11.1、信赖域策略优化算法TRPO强化学习-从理论到实践

基于LunarLander登陆器的TRPO强化学习&#xff08;含PYTHON工程&#xff09; TRPO强化学习算法主要分为3个部分&#xff0c;分别介绍其理论、细节、实现 本文主要介绍TRPO的特点、思路、和优化式子的推导 TRPO系列&#xff08;TRPO是真的复杂&#xff0c;全部理解花费了我半…

【论文笔记】End-to-End Diffusion Latent Optimization Improves Classifier Guidance

Abstract Classifier guidance为图像生成带来了控制&#xff0c;但是需要训练新的噪声感知模型(noise-aware models)来获得准确的梯度&#xff0c;或使用最终生成的一步去噪近似&#xff0c;这会导致梯度错位(misaligned gradients)和次优控制(sub-optimal control)。 梯度错位…

算法训练营Day37

#Java #动态规划 Feeling and experiences&#xff1a; 目标和&#xff1a;力扣题目链接 给你一个非负整数数组 nums 和一个整数 target 。 向数组中的每个整数前添加 或 - &#xff0c;然后串联起所有整数&#xff0c;可以构造一个 表达式 &#xff1a; 例如&#xff0c…

facebook广告素材制作要注意哪些

在制作Facebook广告素材时&#xff0c;需要注意以下几点&#xff1a; 目标受众&#xff1a;了解目标受众的喜好、需求和兴趣&#xff0c;以便制作能够吸引他们的广告素材。广告格式&#xff1a;选择适合广告内容的格式&#xff0c;如图片、视频、幻灯片等。同时&#xff0c;要…

jmeter--1.安装

目录 1. jmeter和jdk下载地址 2. jdk环境变量配置 3. Jmeter的环境变量配置 1. jmeter和jdk下载地址 链接&#xff1a;https://pan.baidu.com/s/1C8ItsR8ZUGJO9uyTJ-9QtQ?pwdx5m5 提取码&#xff1a;x5m5 2. jdk环境变量配置 2.1 我的电脑右键属性-->高级系统设置--&…

20240111在ubuntu20.04.6下解压缩RAR格式的压缩包

20240111在ubuntu20.04.6下解压缩RAR格式的压缩包 2024/1/11 18:25 百度搜搜&#xff1a;ubuntu rar文件怎么解压 rootrootrootroot-X99-Turbo:~/temp$ ll total 2916 drwx------ 3 rootroot rootroot 4096 1月 11 18:28 ./ drwxr-xr-x 25 rootroot rootroot 4096 1月…

详细分析Java中的分布式任务调度框架 XXL-Job

目录 前言1. 基本知识2. Demo3. 实战 前言 可视化任务调度 可视化配置 1. 基本知识 在Java中&#xff0c;分布式任务调度框架 XXL-Job 是一个开源的分布式任务调度平台&#xff0c;用于实现分布式系统中的定时任务调度和分布式任务执行。 下面是关于XXL-Job的一些概念、功…

05- OpenCV:图像操作和图像混合

目录 一、图像操作 1、读写图像 2、读写像素 3、修改像素值 4、Vec3b与Vec3F 5、相关的代码演示 二、图像混合 1、理论-线性混合操作 2、相关API(addWeighted) 3、代码演示&#xff08;完整的例子&#xff09; 一、图像操作 1、读写图像 &#xff08;1&#xff09;…

IDC机房服务器搬迁之运行了几年的服务器没关过机,今天关机下架,再上架突然起不来了,怎么快速处理?

环境 戴尔R420 服务器 1U 2台直连存储 4U CentOS 7 问题描述 IDC机房服务器搬迁之运行了几年的服务器没关过机,今天关机下架,再上架突然起不来了,怎么快速处理? 服务器上电开机就出现进入紧急模式 Welcome to emergency mode! After logging in, type “journalctl …

MYSQL篇--锁机制高频面试题

Mysql锁机制 1对mysql的锁有了解吗&#xff1f; 首先我们要知道&#xff0c;mysql的锁 其实是为了解决在并发事务时所导致的数据不一致问题的一种处理机制&#xff0c;也就是说 在事务的隔离级别实现中&#xff0c;就需要利用锁来解决幻读问题 然后我们可以聊到锁的分类 按锁…