Java 查找具有非整数小时偏移量的时区,用户希望查找所有具有非整数小时偏移量(non-integral hour offset)的时区,获取每个时区的时区偏移量,并计算总秒数除以 3600 之后的剩余时间。
Java 查找具有非整数小时偏移量的时区 问题描述
用户希望查找所有具有非整数小时偏移量(non-integral hour offset)的时区。
Java 查找具有非整数小时偏移量的时区 解决方案
获取每个时区的时区偏移量,并计算总秒数除以 3600 之后的剩余时间。
Java 查找具有非整数小时偏移量的时区 具体实例
大部分时区的 UTC 偏移量为小时的整数。例如,通常所说的北美东部时区(EST)为 UTC-05:00
,而欧洲中部时间(CET)为 UTC+01:00
。不过也存在 UTC 偏移量为半小时甚至 45 分钟的时区,如印度标准时间(IST)为 UTC+05:30
,而查塔姆标准时间(CHAST)为 UTC+12:45
。本范例将讨论利用 java.time
包查找所有具有非整数小时偏移量的时区。
如例 8-33 所示,我们通过 ZoneOffset
类查找每个时区 ID 相对于 UTC 的偏移量,并将其总秒数与 3600 秒(1 小时)进行比较。
例 8-33 查找每个时区 ID 的偏移量(以秒为单位)
public class FunnyOffsets {
public static void main(String[] args) {
Instant instant = Instant.now();
ZonedDateTime current = instant.atZone(ZoneId.systemDefault());
System.out.printf("Current time is %s%n%n", current);
System.out.printf("%10s %20s %13s%n", "Offset", "ZoneId", "Time");
ZoneId.getAvailableZoneIds().stream()
.map(ZoneId::of) ➊
.filter(zoneId -> {
ZoneOffset offset = instant.atZone(zoneId).getOffset(); ➋
return offset.getTotalSeconds() % (60 * 60) != 0; ➌
})
.sorted(comparingInt(zoneId ->
instant.atZone(zoneId).getOffset().getTotalSeconds()))
.forEach(zoneId -> {
ZonedDateTime zdt = current.withZoneSameInstant(zoneId);
System.out.printf("%10s %25s %10s%n",
zdt.getOffset(), zoneId,
zdt.format(DateTimeFormatter.ofLocalizedTime(
FormatStyle.SHORT)));
});
}
}
❶ 将地区 ID(字符串)映射到时区 ID
❷ 计算偏移量
❸ 仅返回偏移量无法被 3600 整除的时区 ID
ZoneId.getAvailableZoneIds
静态方法返回 Set<String>
,表示所有可用的时区 ID;ZoneId.of
方法将生成的字符串流转换为 ZoneId 实例流。
在本例中,筛选器中的 lambda 表达式首先将 atZone
方法应用到 Instant
以创建 ZonedDateTime
,然后应用 getOffset
方法。最后,利用 ZoneOffset
类定义的 getTotalSeconds
方法获取时区偏移量(以秒为单位)。根据 Javadoc 的描述,getTotalSeconds
方法是“访问偏移量的主要方式,它返回小时、分、秒字段的总和(以秒为单位),作为一个偏移量添加到给定的时间”。仅当总秒数无法被 3600(60 sec/min * 60 min/hour
)整除时,筛选器中的 Predicate
才返回 true
。
在打印结果前,程序对生成的 ZoneId
实例排序。sorted
方法传入 Comparator
作为参数。本例使用 Comparator
接口定义的静态方法 comparingInt
,它生成一个根据给定整数键排序的 Comparator
。程序同样采用 getTotalSeconds
方法获取时区偏移量(以秒为单位),ZoneId
实例根据偏移量进行排序。
接下来,针对每个 ZoneId
,程序使用 withZoneSameInstant
方法计算默认时区中当前的 ZonedDateTime
,以打印结果。打印的字符串将显示偏移量、时区 ID 以及相应时区中经过格式化的本地时间。
程序的执行结果如例 8-34 所示。
例 8-34 具有非整数小时偏移量的时区
Current time is 2016-08-08T23:12:44.264-04:00[America/New_York]
Offset ZoneId Time
-09:30 Pacific/Marquesas 5:42 PM
-04:30 America/Caracas 10:42 PM
-02:30 America/St_Johns 12:42 AM
-02:30 Canada/Newfoundland 12:42 AM
+04:30 Iran 7:42 AM
+04:30 Asia/Tehran 7:42 AM
+04:30 Asia/Kabul 7:42 AM
+05:30 Asia/Kolkata 8:42 AM
+05:30 Asia/Colombo 8:42 AM
+05:30 Asia/Calcutta 8:42 AM
+05:45 Asia/Kathmandu 8:57 AM
+05:45 Asia/Katmandu 8:57 AM
+06:30 Asia/Rangoon 9:42 AM
+06:30 Indian/Cocos 9:42 AM
+08:45 Australia/Eucla 11:57 AM
+09:30 Australia/North 12:42 PM
+09:30 Australia/Yancowinna 12:42 PM
+09:30 Australia/Adelaide 12:42 PM
+09:30 Australia/Broken_Hill 12:42 PM
+09:30 Australia/South 12:42 PM
+09:30 Australia/Darwin 12:42 PM
+10:30 Australia/Lord_Howe 1:42 PM
+10:30 Australia/LHI 1:42 PM
+11:30 Pacific/Norfolk 2:42 PM
+12:45 NZ-CHAT 3:57 PM
+12:45 Pacific/Chatham 3:57 PM
可以看到,将 java.time
包中的多个类结合在一起使用,就能解决复杂而有趣的问题。