Java时区国际化解决方案

在这里插入图片描述
当用户所在时区和服务器所在时区不一致时,会产生时区相关问题,如时间显示错误、程序取得的时间和数据库存储的时间不一致、定时任务的触发没有跟随用户当前的时区等等问题.

统一拦截时区

/*****/
@Component
@Slf4j
public class TimeZoneIdInterceptor implements HandlerInterceptor {private static final String TIME_ZONE_CODE = "timeZoneCode";/*** 在控制器方法处理之前执行的预处理方法* <p>* 该方法用于在控制器方法执行前进行一些预处理操作,例如检查用户是否登录、记录日志等 如果该方法返回false,则表示不执行控制器方法;如果返回true,则表示继续执行控制器方法** @param request  请求对象,用于获取请求信息* @param response 响应对象,用于向客户端发送响应* @param handler  控制器方法的对象,用于执行具体的控制器方法* @return 是否执行控制器的处理逻辑,true表示执行,false表示不执行* @throws Exception 如果发生异常,将抛出异常*/@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)throws Exception {//获取TimeZoneId 默认东八区String timeZoneId=request.getHeader(TIME_ZONE_CODE);if(StringUtilsPlus.isBlank(timeZoneId)){timeZoneId = ZoneIdEnum.CTT.getZoneIdName();}TimeZoneIdContext.setTimeZoneId(timeZoneId);return true;}/*** controller方法执行后* <p>* 该方法主要用于在控制器方法处理完请求后进行一些额外的处理或清理工作 它是拦截器中的一部分,用于在请求被控制器处理后立即执行自定义逻辑** @param request      请求对象,用于访问请求信息* @param response     响应对象,用于控制响应给客户端的信息* @param handler      被执行的控制器方法的引用,用于识别和可能操作控制器方法* @param modelAndView ModelAndView对象,包含要渲染的视图和模型数据* @throws Exception 如果在处理过程中发生异常,可以将其抛出*/@Overridepublic void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,ModelAndView modelAndView) throws Exception {log.info("postHandle执行{}", modelAndView);TimeZoneIdContext.removeTimeZoneId();}}

GET请求及POST表单请求(RequestParam和PathVariable参数):
自定义spring mvc的参数解析器,配置Converter<String, T>转换器实现参数转换

@Configuration
@Slf4j
public class DateConverterConfig {/*** 创建一个字符串到LocalDateTime的转换器** 该方法通过定义一个Converter接口的匿名类来实现字符串到LocalDateTime的对象转换* 主要用于Spring框架中,以便在数据绑定、类型转换时使用** @return Converter<String, LocalDateTime> 返回一个实现了Converter接口的匿名类实例,*         用于将字符串转换为LocalDateTime对象*/@Beanpublic Converter<String, LocalDateTime> localDateTimeConverter() {return new Converter<String, LocalDateTime>() {@Overridepublic LocalDateTime convert(String source) {// String类型的日期字符串转为LocalDateTime类型,并加上时区处理// 如果源字符串不为空,则进行转换;否则返回nullif(StringUtilsPlus.isNotBlank(source)){return TimeZoneUtils.convertTimeZoneLocalDateTime(source, TimeZoneIdContext.getTimeZoneId().get(),ZoneIdEnum.CTT.getZoneIdName());}return null;}};}/*** 创建一个字符串到日期的转换器** 该方法通过定义一个Converter的Bean,实现了从字符串类型到Date类型的转换主要用于处理日期字符串,* 并考虑了时区的转换** @return Converter<String, Date> 实现了从字符串到日期的转换功能*/@Beanpublic Converter<String, Date> dateConverter() {return new Converter<String, Date>() {@Overridepublic Date convert(String source) {// String类型的日期字符串转为LocalDateTime类型,并加上时区处理if(StringUtilsPlus.isNotBlank(source)){return TimeZoneUtils.convertTimeZoneDate(source, TimeZoneIdContext.getTimeZoneId().get(),ZoneIdEnum.CTT.getZoneIdName());}return null;}};}
}
  1. POST-application/json请求(RequestBody参数)在使用javabean作为入参时,javabean对象中的Date、LocalDateTime类型可以根据请求头中的时区字段自动转为应用服务当前时区的时间
  2. 接口返回对象时,对象中的Date、LocalDateTime类型的日期值,可以根据请求头中的时区字段,自动转为该时区的时间
@JsonComponent
public class DateJsonSerializerConfig {/*** 反序列化,其它时区的时间转为本地时间*/public static class LocalDateTimeJsonDeserializer extends JsonDeserializer<LocalDateTime> {@Overridepublic LocalDateTime deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException {// 根据时区字段将日期转为本地时区时间// String类型日期转为LocalDateTime类型String timeZone = StringUtilsPlus.defaultIfEmpty(TimeZoneIdContext.getTimeZoneId().get(), ZoneIdEnum.CTT.getZoneIdName());String value  = jsonParser.getValueAsString();if(StringUtilsPlus.isNotBlank(value)){return TimeZoneUtils.convertTimeZoneLocalDateTime(value,timeZone,ZoneIdEnum.CTT.getZoneIdName());}return null;}}/*** 序列化,本地时间转为其它时区的时间*/public static class LocalDateTimeJsonSerializer extends JsonSerializer<LocalDateTime> {@Overridepublic void serialize(LocalDateTime localDateTime, JsonGenerator jsonGenerator,SerializerProvider serializerProvider) throws IOException {//  本地时间转对应时区的时间String timeZone = StringUtilsPlus.defaultIfEmpty(TimeZoneIdContext.getTimeZoneId().get(), ZoneIdEnum.CTT.getZoneIdName());if(localDateTime!=null) {jsonGenerator.writeString(TimeZoneUtils.convertTimeZoneLocalDateTime(localDateTime, timeZone).format(DateTimeFormatter.ofPattern(GlobalConstants.YYYY_MM_DD_HH_MM_SS)));}}}/*** 反序列化,其它时区的时间转为本地时间*/public static class DateTimeJsonDeserializer extends JsonDeserializer<Date> {@Overridepublic Date deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException {// 根据时区字段将日期转为本地时区时间// String类型日期转为LocalDateTime类型String timeZone = StringUtilsPlus.defaultIfEmpty(TimeZoneIdContext.getTimeZoneId().get(), ZoneIdEnum.CTT.getZoneIdName());String value  = jsonParser.getValueAsString();if(StringUtilsPlus.isNotBlank(value)){return TimeZoneUtils.convertTimeZoneDate(value,timeZone,ZoneIdEnum.CTT.getZoneIdName());}return null;}}/*** 序列化,本地时间转为其它时区的时间*/public static class DateTimeJsonSerializer extends JsonSerializer<Date> {@Overridepublic void serialize(Date date, JsonGenerator jsonGenerator,SerializerProvider serializerProvider) throws IOException {//  本地时间转对应时区的时间String timeZone = StringUtilsPlus.defaultIfEmpty(TimeZoneIdContext.getTimeZoneId().get(), ZoneIdEnum.CTT.getZoneIdName());if(date!=null) {jsonGenerator.writeString(DateUtil.format(TimeZoneUtils.convertTimeZoneDate(date, timeZone),GlobalConstants.YYYY_MM_DD_HH_MM_SS));}}}
}

时间时区相互转化工具类

/*** 时间时区转化**/
public class TimeZoneUtils {/**** 时间转化* 将给定的日期对象从一个时区转换到另一个时区* @param date 需要转换的日期对象* @param timeZone 目标时区* @return 转换后的日期对象,如果目标时区与默认时区相同,则返回原始日期对象*/public static Date convertTimeZoneDate(Date date, String timeZone) {// 检查目标时区是否为默认时区,如果是,则无需转换,直接返回原始日期if (isDefaultTimeZone(timeZone)) {return date;}// 获取目标时区TimeZone otherZone = TimeZone.getTimeZone(timeZone);// 创建SimpleDateFormat对象,用于日期的格式化和解析SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");// 设置SimpleDateFormat的时区为其他时区sdf.setTimeZone(otherZone);// 用于存储转换后的日期对象Date otherZoneDate = null;try {// 将格式化后的日期字符串解析为日期对象,该对象现在处于目标时区// 使用纽约时区格式化日期String dateInNewYork = sdf.format(date);otherZoneDate = DateUtil.parse(dateInNewYork,"yyyy-MM-dd HH:mm:ss");} catch (Exception e) {// 如果发生异常,打印异常信息e.printStackTrace();}// 返回转换后的日期对象return otherZoneDate;}/*** 将LocalDate对象转换为特定时区的LocalDate对象 此方法主要用于在处理不同时区的日期时进行转换,以确保日期的正确解析和表示** @param localDate 输入的LocalDate对象,表示日期* @param timeZone  目标时区的字符串表示,如"America/New_York"* @return 转换后特定时区的LocalDate对象*/public static LocalDate convertTimeZoneLocalDate(LocalDate localDate, String timeZone) {if (isDefaultTimeZone(timeZone)) {return localDate;}// 指定时区ZoneId zoneId = ZoneId.of(timeZone);// 将LocalDate转换为指定时区的ZonedDateTime 该时区一天的开始时刻的ZonedDateTime zonedDateTime = localDate.atStartOfDay(zoneId);// 返回转换后的LocalDate对象return zonedDateTime.toLocalDate();}/*** 将给定的时间字符串从一个时区转换到另一个时区* 此方法首先将时间字符串解析为本地日期时间对象,然后将其转换到源时区,* 最后将时间转换到目标时区,保持时间点不变** @param dateTimeString 时间字符串,表示源时区中的本地日期和时间* @param sourceTimeZone   源时区的ID,例如"America/New_York"* @param targetTimeZone   目标时区的ID,例如"Europe/London"* @return 转换到目标时区后的本地日期时间对象*/public static LocalDateTime convertTimeZoneLocalDateTime(String dateTimeString,String sourceTimeZone,String targetTimeZone) {// 获取源时区的ZoneId对象ZoneId sourceZoneId = ZoneId.of(sourceTimeZone);// 获取目标时区的ZoneId对象ZoneId targetZoneId = ZoneId.of(targetTimeZone);// 创建DateTimeFormatter指定日期时间格式DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss").withZone(sourceZoneId);// 解析字符串到ZonedDateTimeZonedDateTime zonedDateTimeSource = ZonedDateTime.parse(dateTimeString, formatter);// 转换到另一个时区ZonedDateTime zonedDateTimeTarget = zonedDateTimeSource.withZoneSameInstant(targetZoneId);// 将源时区应用到本地日期时间,并转换到目标时区,最后返回转换后的本地日期时间return zonedDateTimeTarget.toLocalDateTime();}/*** 转换日期时间字符串从一个时区到另一个时区* 此方法首先根据源时区解析输入的日期时间字符串,然后将其转换到目标时区* 最后返回转换后的日期时间的Date对象** @param dateTimeString 日期时间字符串,格式为"yyyy-MM-dd HH:mm:ss"* @param sourceTimeZone 源时区ID,如"Asia/Shanghai"* @param targetTimeZone 目标时区ID,如"America/New_York"* @return 转换时区后的日期时间对象*/public static Date convertTimeZoneDate(String dateTimeString,String sourceTimeZone,String targetTimeZone) {// 获取源时区的ZoneId对象ZoneId sourceZoneId = ZoneId.of(sourceTimeZone);// 获取目标时区的ZoneId对象ZoneId targetZoneId = ZoneId.of(targetTimeZone);// 创建DateTimeFormatter指定日期时间格式DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss").withZone(sourceZoneId);// 解析字符串到ZonedDateTimeZonedDateTime zonedDateTimeSource = ZonedDateTime.parse(dateTimeString, formatter);// 转换到另一个时区ZonedDateTime zonedDateTimeTarget = zonedDateTimeSource.withZoneSameInstant(targetZoneId);// 将源时区应用到本地日期时间,并转换到目标时区,最后返回转换后的本地日期时间LocalDateTime targetLocalDateTime = zonedDateTimeTarget.toLocalDateTime();Date date = Date.from(targetLocalDateTime.atZone(targetZoneId).toInstant());return date;}/*** 将LocalDateTime对象从默认时区转换到指定时区** @param localDateTime 需要转换的LocalDateTime对象* @param timeZone      目标时区的ID字符串* @return 转换后的LocalDateTime对象*/public static LocalDateTime convertTimeZoneLocalDateTime(LocalDateTime localDateTime, String timeZone) {if (isDefaultTimeZone(timeZone)) {return localDateTime;}// 默认时区ZoneId sourceZoneId = currentZoneId();ZonedDateTime sourceZonedDateTime = localDateTime.atZone(sourceZoneId);// 新时区ZoneId targetZoneId = ZoneId.of(timeZone);ZonedDateTime targetZonedDateTime = sourceZonedDateTime.withZoneSameInstant(targetZoneId);// 时区转换return targetZonedDateTime.toLocalDateTime();}/*** 是否是默认时区** @param timeZone* @return*/public static boolean isDefaultTimeZone(String timeZone) {return StringUtilsPlus.isBlank(timeZone) || currentZoneId().getId().toUpperCase().equals(timeZone.toUpperCase());}/*** 指定默认时区** @return java.time.ZoneId 返回指定的时区ID*/public static ZoneId specifyDefaultZone() {// 创建一个时区ID对象,指定时区为中国的北京时间ZoneId zoneId = ZoneId.of(ZoneIdEnum.CTT.getZoneIdName());// 返回创建的时区ID对象return zoneId;}/*** 获取当前系统的时区ID* <p>* 此方法用于识别和返回系统默认的时区ID它在需要时区信息来处理日期和时间的场景中特别有用** @return ZoneId 当前系统的时区ID*/public static ZoneId currentZoneId() {// 获取系统默认时区的时区IDZoneId systemDefaultZone = ZoneId.systemDefault();return systemDefaultZone;}/*** 指定时区获取当前日期* 此方法主要解决在不同地区运行代码时获取的日期不一致的问题* 通过指定默认时区,确保在任何地方运行都能获取到正确的日期** @return 返回指定时区的当前日期*/public static LocalDate specifyTimeZoneLocalDate() {return LocalDate.now(specifyDefaultZone());}/*** 获取指定时区的当前日期和时间* 此方法通过指定默认时区来获取当前的日期和时间,主要用于需要考虑时区的因素以确保时间一致性的场景** @return 返回指定时区的当前日期和时间*/public static LocalDateTime specifyTimeZoneLocalTimeDate() {return LocalDateTime.now(specifyDefaultZone());}/*** 获取  亚洲/上海 当前时间** 此方法旨在提供一个简单的方式来获取当前时间,并将其转换为CTT时区的时间* 它使用了更通用的convertTimeZoneLocalDateTime方法来执行实际的时区转换** @return 当前时间转换为CTT时区后的LocalDateTime*/public static LocalDateTime cttLocalDateTime() {return convertTimeZoneLocalDateTime(LocalDateTime.now(), ZoneIdEnum.CTT.getZoneIdName());}
}

参考:链接

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

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

相关文章

前端开发设计模式——状态模式

目录 一、状态模式的定义和特点 二、状态模式的结构与原理 1.结构&#xff1a; 2.原理&#xff1a; 三、状态模式的实现方式 四、状态模式的使用场景 1.按钮的不同状态&#xff1a; 2.页面加载状态&#xff1a; 3.用户登录状态&#xff1a; 五、状态模式的优点 1.提…

RabbitMQ 入门(七)SpringAMQP五种消息类型(Topic Exchange)

一、Topic Exchange&#xff08;消息模式&#xff09; TopicExchange 与DirectExchange类似&#xff0c;区别在于routingKey可以是多个单词的列表&#xff0c;并且以.分割。 Topic类型的Exchange与Direct相比&#xff0c;都是可以根据RoutingKey把消息路由到不同的队列。只不过…

数据结构与算法——Java实现 42.二叉树的最大深度

苦尽甘来时&#xff0c;一路向阳开 —— 24.10.21 104. 二叉树的最大深度 给定一个二叉树 root &#xff0c;返回其最大深度。 二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&a…

微软数据恢复工具- “快速扫描” 和 “深度扫描” 两种模式 快速扫描的速度更快,使用 NTFS 文件系统下的目录结构

提供了 “快速扫描” 和 “深度扫描” 两种模式。快速扫描的速度更快&#xff0c;使用 NTFS 文件系统下的目录结构和文件名恢复文件&#xff1b;而深度扫描则能帮你恢复更多丢失目录结构和文件。有了 WinFR 界面版&#xff0c;你不需要再学习任何复杂的命令行操作了&#xff0c…

extra_model_paths.yaml解读

为了将模型文件放置在1个共享位置&#xff0c;以方便重装comfyui或其他需要用到模型共享的情况&#xff0c;将在修改extra_model_paths.yaml中遇到的错误情况汇总如下&#xff1a; 1、当模型路径指引前面空格不是4个时错误如下&#xff08;示例范本中后面的例子就是因为是5个空…

重磅揭秘,AI 编程崛起,真的会让程序员面临裁员危机吗?

"完了&#xff0c;AI 要取代程序员了&#xff01;" 我的朋友圈里经常会分享一些 AI、AI 编程的东西&#xff0c;最近收到不少人的私信&#xff1a; "要不要转行啊&#xff1f;""现在学编程还有意义吗&#xff1f;""听说隔壁公司已经用 AI…

117. 填充每个节点的下一个右侧节点指针 II【 力扣(LeetCode) 】

文章目录 零、LeetCode 原题一、题目描述二、测试用例三、解题思路3.1 层次遍历3.2 层次遍历&#xff08;优化&#xff09; 四、参考代码4.1 层次遍历4.2 层次遍历&#xff08;优化&#xff09; 零、LeetCode 原题 117. 填充每个节点的下一个右侧节点指针 II 一、题目描述 给…

OpenCV高级图形用户界面(17)设置一个已经创建的滚动条的最小值函数setTrackbarMin()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 cv::setTrackbarMin 这个函数的作用就是设置指定窗口中轨迹条的最小位置。这使得开发者能够在程序运行时动态地调整轨迹条的范围&#xff0c;而不…

如何安装和初始化飞牛私有云 fnOS?

如何安装和初始化飞牛私有云 fnOS&#xff1f;

万家数科:零售业务信息化融合的探索|OceanBase案例

本文作者&#xff1a;马琳&#xff0c;万家数科数据库专家。 万家数科商业数据有限公司&#xff0c;作为华润万家旗下的信息技术企业&#xff0c;专注于零售行业&#xff0c;在为华润万家提供服务的同时&#xff0c;也积极面向市场&#xff0c;为零售商及其生态系统提供全面的核…

对称二叉树

给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 示例 1&#xff1a; 输入&#xff1a;root [1,2,2,3,4,4,3] 输出&#xff1a;true示例 2&#xff1a; 输入&#xff1a;root [1,2,2,null,3,null,3] 输出&#xff1a;false提示&#xff1a; 树中节点数目在范围…

一款实现PLC扩展CANFD的好工具 — PXB-6020D协议转换器

如何轻松实现PLC扩展CAN FD&#xff1f;本文将简单介绍PLC上的CAN接口&#xff0c;并分享一款简单的好工具——PXB-6020D&#xff0c;它能帮助我们轻松实现从Modbus到CANFD的无缝转换。 在工业自动化领域&#xff0c;PLC&#xff08;可编程逻辑控制器&#xff09;是核心组件之一…

民宿在线预订:SpringBoot技术实践指南

2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统&#xff0c;它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等&#xff0c;非常…

基于SSM服装定制系统的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;用户管理&#xff0c;服装类型管理&#xff0c;服装信息管理&#xff0c;服装定制管理&#xff0c;留言反馈&#xff0c;系统管理 前台账号功能包括&#xff1a;系统首页&#xff0c;个人中心&#xf…

Linux LCD 驱动实验

LCD 是很常用的一个外设&#xff0c;在裸机篇中我们讲解了如何编写 LCD 裸机驱动&#xff0c;在 Linux 下LCD 的使用更加广泛&#xff0c;再搭配 QT 这样的 GUI 库下可以制作出非常精美的 UI 界面。本章我们就来学习一下如何在 Linux 下驱动 LCD 屏幕。 Framebuffer 设备 先来…

基于vue框架的的点餐系统1o2te(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。

系统程序文件列表 项目功能&#xff1a;用户,商家,菜品分类,菜品信息 开题报告内容 基于Vue框架的点餐系统开题报告 一、研究背景与意义 随着移动互联网技术的飞速发展&#xff0c;餐饮行业也迎来了数字化转型的浪潮。传统的点餐方式&#xff0c;如纸质菜单和人工记录&…

Linux系统基础-文件系统

个人主页&#xff1a;C忠实粉丝 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 C忠实粉丝 原创 Linux系统基础-文件系统 收录于专栏[Linux学习] 本专栏旨在分享学习Linux的一点学习笔记&#xff0c;欢迎大家在评论区交流讨论&#x1f48c; 目录 1. 回顾C语言…

【排序】——1.冒泡排序法(含优化)

冒泡排序 1.原理 左边大于右边交换一趟排下来最大的交换到右边来(接下来所以文章用升序举例) 从左到右&#xff0c;相邻元素进行比较。 每次比较一轮&#xff0c;就会找到序列中最大的一个&#xff08;最小的一个——降序&#xff09;。这个数就会从序列的最右边冒出来。 以…

《Spring Cloud Config与Bus整合实现微服务配置自动刷新》

目录 Config与Bus整合自动刷新步骤1&#xff1a;安装RabbitMQ并启动RabbitMQ的安装 步骤2&#xff1a;创建项目创建Eureka Server创建config-server 步骤3&#xff1a; 添加依赖步骤4&#xff1a;Config Client步骤5&#xff1a;测试运行问题一问题二 总结 Config与Bus整合自动…

SQL Server-导入和导出excel数据-注意事项

环境&#xff1a; win10&#xff0c;SQL Server 2008 R2 之前写过的放在这里&#xff1a; SqlServer_陆沙的博客-CSDN博客 https://blog.csdn.net/pxy7896/category_12704205.html 最近重启ASP.NET项目&#xff0c;在使用sql server导出和导入数据时遇到一些问题&#xff0c;特…