JVM默认时区为:Asia/Shanghai与java程序中GMT+08不一致异常

 更新时间:2022年10月31日 10:27:36   作者:ACoderForever  
这篇文章主要介绍了JVM默认时区为:Asia/Shanghai与java程序中GMT+08不一致异常问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

在Spring程序中配置了spring.jackson.time-zone=GMT+08时,部分时间相差一个小时问题,且是固定的时间出现了固定的时差问题。

经过排查,发现是JVM的默认时区为

Asia/Shanghai,两者不一致,然后Asia/Shanghai 这个时区并不一定与GMT+08这个时区相等,他们是2种定义标准。

Asia/Shanghai 这个代表的是中国的时区,但在历史中,有国家(包含中国)政策颁布了在1986-1991年等还存在夏令时。

在这样的时间区间,夏季时,会将时间拨快1个小时(即东9区时间),夏季结束时会再次将时间拨回一个小时(即东8区时间)。

所以要保证程序显示的时间没有问题,需要将JVM和spring.jackson.time-zone设置的时区保持一致即可解决问题

JVM中设置为

Asia/Shanghai,经代码调试出现的底层时区调整的测试案例。

 import java.text.SimpleDateFormat;
import java.util.Date;

/**
 * @author Ashen
 * @date 16/07/2019
 */
public class Test {

	public static SimpleDateFormat sdf = new SimpleDateFormat("yyyy.MM.dd HH:mm:ss");

	static int[] offsets = {
			28800000,
			29143000,
			32400000,
			3600000
	};
	public static void main(String[] args) {
		test2();
	}

	public static void test1(){
		Date date0 = new Date(1986 - 1900, 07 - 1, 29);
		Date date1 = new Date(1980 - 1900, 07 - 1, 29);
		Date date2 = new Date(1988 - 1900, 07 - 1, 29);
	}

	public static void test2() {
		// 以下为JVM中时区为:Asia/Shanghai时的偏移量参数值(共9对,分别为9年夏令时的开启与结束点)
		long transitions[] = {
				-9048018124799999L,
				-8918966038528000L,
				-3823593062399950L,
				-3781140480000000L,
				-3722379263999950L,
				-3651969024000000L,
				2111569920000050L,
				2158623129600000L,
				2232955699200050L,
				2287440691200000L,
				2361773260800050L,
				2416258252800000L,
				2493068083200050L,
				2547553075200000L,
				2621885644800050L,
				2676370636800000L,
				2750703206400050L,
				2805188198400000L,
				8660385792000000L
		};
		for(int i=0;i<18;i++){
			Long longTime = transitions[i] >> 12;
			Long field2 = transitions[i] << 52 >> 60;
			Long field3 = transitions[i] << 56 >> 60;
			Long field4 = transitions[i] << 60 >> 60;
			Date date = new Date(longTime);
			System.out.println("时间"+getIndexStr(i)+": "+sdf.format(date)+" 保留值>>"+field2+" 日光节约量 >> "+getOffset(field3)+" GMT偏移量 >> "+getOffset(field4));
		}

	}

	public static String getIndexStr(int i){
		if(i<10) return "0"+i;
		else return String.valueOf(i);
	}

	public static String getOffset(long offset){
		return offsets[(int)offset]/(1000 *60* 60)+"小时,";
	}

}

下面为运行结果:清晰的看到Asia/Shanghai 时区的调整时间点与调整量。

transitions[] 中的值来源于DEBUG程序时复制jvm中的数据。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • eclipse启动出现“failed to load the jni shared library”问题解决

    eclipse启动出现“failed to load the jni shared library”问题解决

    这篇文章主要介绍了eclipse启动出现“failed to load the jni shared library”问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-11-11
  • Java中String类的常见方法超详细讲解

    Java中String类的常见方法超详细讲解

    这篇文章主要介绍了Java中String类常见方法的相关资料,String类是不可变的,字符串常量池用于存储字符串字面量,常用方法包括字符串查找、转换、比较、替换、拆分和截取,需要的朋友可以参考下
    2025-04-04
  • 图文详解Maven工程打jar包的N种方式

    图文详解Maven工程打jar包的N种方式

    最近在打包maven项目时,该项目中仅有一个测试类,想打成jar包运行测试,所以下面这篇文章主要给大家介绍了关于Maven工程打jar包的N种方式,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-09-09
  • Java利用策略模式优化过多if else代码

    Java利用策略模式优化过多if else代码

    这篇文章主要介绍了Java利用策略模式优化过多if else代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-09-09
  • SpringBoot 日志的配置及输出应用教程

    SpringBoot 日志的配置及输出应用教程

    Spring Boot 默认使用 SLF4J+Logback 记录日志,并提供了默认配置。本文我们将重点介绍Spring Boot日志的配置及输出。感兴趣的小伙伴可以了解一下
    2021-12-12
  • SpringCloud的网关Zuul和Gateway详解

    SpringCloud的网关Zuul和Gateway详解

    SpringCloudZuul和SpringCloudGateway都是用于构建微服务架构中的API网关的组件,但SpringCloudGateway在性能、功能特性和生态支持等方面有一些优势,因此推荐使用SpringCloudGateway作为首选
    2025-02-02
  • 使用@RequestParam设置默认可以传空值

    使用@RequestParam设置默认可以传空值

    这篇文章主要介绍了使用@RequestParam设置默认可以传空值的操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-08-08
  • SpringBoot CommandLineRunner应用启动后执行代码实例

    SpringBoot CommandLineRunner应用启动后执行代码实例

    本文将深入探讨CommandLineRunner的工作原理、使用场景及最佳实践,帮助开发者充分利用这一功能,构建更加健壮的Spring Boot应用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2025-04-04
  • Java SpringBoot整合SpringCloud

    Java SpringBoot整合SpringCloud

    SpringCloud专注于为典型的用例和扩展机制提供良好的开箱即用体验,今天小编就带大家认识SpringCloud都有些什么特点,感兴趣的小伙伴留下来阅读全文吧
    2021-09-09
  • 老生常谈Java 网络编程 —— Socket 详解

    老生常谈Java 网络编程 —— Socket 详解

    这篇文章主要介绍了Java 网络编程 —— Socket 相关知识,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-05-05

最新评论