TimeZone是Java处理时区的抽象类,需通过getTimeZone()等静态方法获取实例,提供偏移量、夏令时规则及时区ID等元数据,配合Calendar或ZonedDateTime完成时间转换;推荐使用java.time包替代。

TimeZone 类是 Java 中处理时区的核心工具,但它本身是抽象类,不能直接 new,必须通过静态方法获取实例。 它不负责时间计算,主要作用是提供时区偏移量(如 UTC+8)、夏令时规则、时区 ID(如 "Asia/Shanghai")等元数据,配合 Calendar 或 ZonedDateTime(Java 8+)完成真实的时间转换。
获取常用时区实例
最常用的是通过 TimeZone.getTimeZone(String id) 获取。ID 可以是标准区域名(推荐),也可以是缩写或偏移字符串:
-
区域名(强烈推荐):如
TimeZone.getTimeZone("Asia/Shanghai")、TimeZone.getTimeZone("America/New_York")。这类 ID 稳定、支持历史规则(含夏令时变化)。 -
UTC 偏移格式(仅限简单场景):如
TimeZone.getTimeZone("GMT+08:00")或TimeZone.getTimeZone("UTC-05:00")。它生成的是固定偏移时区(FixedOffsetTimeZone),不支持夏令时,也不随系统时区数据库更新。 -
系统默认时区:用
TimeZone.getDefault(),返回 JVM 启动时读取的操作系统时区设置,可被TimeZone.setDefault()修改(但不建议在多线程应用中随意改)。
获取偏移量与夏令时信息
给定一个时间点(毫秒值),可查该时刻的原始偏移(raw offset)和是否启用夏令时(DST):
-
getRawOffset():返回标准时间偏移(单位毫秒),例如上海为28800000(即 +8 小时)。 -
inDaylightTime(Date date):判断指定时间是否处于夏令时期间(对 "Asia/Shanghai" 总是 false,因中国不实行夏令时;对 "Europe/Paris" 则可能返回 true)。 -
getOffset(long time):返回指定时间点的**总偏移量**(= rawOffset + DST 偏移)。例如巴黎冬令时为 +1 小时,夏令时为 +2 小时,此方法会自动返回对应值。
与 Calendar 配合做时区转换(传统方式)
在 Java 7 及以前,常用 TimeZone + Calendar 实现时间显示转换:
立即学习“Java免费学习笔记(深入)”;
- 先创建一个
Calendar实例(如GregorianCalendar),并用setTimeZone()指定时区; - 用
setTimeInMillis()设置统一的时间戳(UTC 毫秒); - 再调用
get(Calendar.HOUR_OF_DAY)等方法,获取该时区下的本地时间字段。
示例:将同一时间戳分别按纽约和上海时区解析:
long now = System.currentTimeMillis();
Calendar ny = new GregorianCalendar(TimeZone.getTimeZone("America/New_York"));
Calendar sh = new GregorianCalendar(TimeZone.getTimeZone("Asia/Shanghai"));
ny.setTimeInMillis(now);
sh.setTimeInMillis(now);
System.out.println("NY: " + ny.get(Calendar.HOUR_OF_DAY) + ":" + ny.get(Calendar.MINUTE));
System.out.println("SH: " + sh.get(Calendar.HOUR_OF_DAY) + ":" + sh.get(Calendar.MINUTE));
注意:优先使用 java.time(Java 8+)
TimeZone 和 Calendar 已被标记为遗留 API。现代项目应使用 java.time 包:
-
ZoneId.of("Asia/Shanghai")替代TimeZone.getTimeZone(...); -
ZonedDateTime封装了时间+时区+夏令时逻辑,比手动算偏移更安全; -
Instant表示 UTC 时间线上的瞬时点,是跨时区转换的可靠基准。
例如:把北京时间转成纽约时间,一行即可完成:ZonedDateTime sh = ZonedDateTime.now(ZoneId.of("Asia/Shanghai"));ZonedDateTime ny = sh.withZoneSameInstant(ZoneId.of("America/New_York"));










