在日常开发中,时区问题常常像潜伏的地雷,平时看似没事,一旦跨地区、跨服务器或跨系统部署,就可能出现日志错乱、数据库时间偏移、定时任务不准等问题。作为程序员,你必须了解 spring boot 应用中 时区配置的各个层面,才能避免凌晨三点被时间问题追着哭。
本文我将从服务器、spring boot、数据库、容器以及前端交互五个方面详细讲解,顺便分享常用时区对照表和踩坑经验。
1. 服务器时区:问题的源头
服务器时区是所有时间问题的起点。很多开发者只关注应用代码,却忽略了服务器系统的时区配置。
- linux 系统:时区信息存放在
/etc/localtime
,可以通过timedatectl
查看和修改。 - windows 系统:使用系统自带时区设置,但在远程服务器或容器中可能与开发环境不一致。
踩坑案例:我曾经遇到过一个定时任务,每天凌晨 0 点在开发机上执行正常,但部署到美国东部的服务器后,实际执行时间变成了前一天的上午 11 点,因为服务器默认时区是 cst(美国中部时间),而开发机是北京时间。
解决方法有两种,一是修改服务器时区,而是修改 jvm 时区。
# 查看当前服务器时区 timedatectl # 设置为北京时间(asia/shanghai) sudo timedatectl set-timezone asia/shanghai
当然很多时候,服务器的时区不是我们能改得,我们就只能通过下面方式修改 jvm 时区。
同时,jvm 启动时也要指定时区:
java -jar -duser.timezone=asia/shanghai app.jar
这样,服务器系统时间和 jvm 默认时间就统一,避免时间偏差。
2. spring boot 全局时区配置
即使服务器时区正确,如果 spring boot 没有显式指定时区,部分组件仍然可能使用 jvm 默认时区,导致日志、定时任务或数据序列化出现偏差。
方法1 全局默认时区
在启动类中设置全局时区:
@springbootapplication public class myapplication { public static void main(string[] args) { // 设置全局默认时区 timezone.setdefault(timezone.gettimezone("asia/shanghai")); springapplication.run(myapplication.class, args); } }
注意:timezone.setdefault
必须在 springapplication.run
前调用,否则应用组件初始化时可能仍使用 jvm 原始时区。
方法2 jackson 序列化时区
spring boot 默认使用 jackson 将 java 日期对象序列化为 json,如果不设置时区,返回给前端的时间可能是 cst(美国时间),导致用户看到的时间不对。
配置方法:
spring: jackson: time-zone: asia/shanghai
或者在代码中:
@bean public objectmapper objectmapper() { objectmapper mapper = new objectmapper(); mapper.settimezone(timezone.gettimezone("asia/shanghai")); return mapper; }
3. 数据库时区:mysql 配置
数据库是时区问题的高发区。mysql 的 timestamp
类型会自动根据服务器时区进行存储转换,如果 jdbc 连接没有指定时区,写入的数据就可能与预期不一致。
方法1 数据库层配置
# 查看当前时区 select @@global.time_zone, @@session.time_zone; # 设置数据库时区为北京时间 set global time_zone = '+08:00'; set time_zone = '+08:00';
方法2 jdbc 层配置
spring boot 配置:
spring: datasource: url: jdbc:mysql://localhost:3306/testdb?servertimezone=asia/shanghai&usessl=false&characterencoding=utf-8 username: root password: 123456
servertimezone
参数指定 jdbc 连接使用的时区,避免报 the server time zone value is unrecognized
错误,并确保写入和读取的时间一致。
踩坑案例:有一次,我的定时任务写入数据库的 localdatetime
,在开发机上显示正确,但生产数据库显示早了 8 小时,原因是生产服务器和 mysql 容器默认时区是 cst(美国时间)。
4. 容器化部署时区配置
在 docker 或 kubernetes 环境中,容器默认使用镜像系统的时区,和宿主机或 jvm 时区可能不同,因此必须显式配置。
方法1 docker 环境变量
environment: - tz=asia/shanghai - java_opts=-duser.timezone=asia/shanghai
方法2 挂载宿主机时区
volumes: - /etc/localtime:/etc/localtime:ro - /etc/timezone:/etc/timezone:ro
优点:确保容器和宿主机时区一致,避免跨容器时间偏差。
方法3 dockerfile 内固定时区
from openjdk:17-jdk-slim env tz=asia/shanghai run ln -snf /usr/share/zoneinfo/$tz /etc/localtime && echo $tz > /etc/timezone copy target/myapp.jar /app.jar entrypoint ["java","-duser.timezone=asia/shanghai","-jar","/app.jar"]
5. 常用时区对照
在 spring boot、数据库、docker 等场景中,推荐使用 iana 时区 id,避免缩写歧义。
中国及东亚
时区 id | 描述 | utc 偏移 |
---|---|---|
asia/shanghai | 北京时间 | utc+08:00 |
asia/urumqi | 乌鲁木齐时间 | utc+06:00 |
asia/tokyo | 日本时间 | utc+09:00 |
北美
时区 id | 描述 | utc 偏移 |
---|---|---|
america/new_york | 美国东部时间 | utc-05:00 / utc-04:00 |
america/chicago | 美国中部时间 | utc-06:00 / utc-05:00 |
欧洲
时区 id | 描述 | utc 偏移 |
---|---|---|
europe/london | 英国时间 | utc+0 / utc+1 |
europe/paris | 法国时间 | utc+1 / utc+2 |
5. 总结
时区配置虽然看似简单,但涉及的层面很多,需要在操作系统、jvm、spring boot应用、数据库等各个层面保持一致。
到此这篇关于深入详解springboot中时区问题解决与配置方案的文章就介绍到这了,更多相关springboot时区配置内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
发表评论