Java 查找具有非整数小时偏移量的时区

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 包中的多个类结合在一起使用,就能解决复杂而有趣的问题。

Python教程

Java教程

Web教程

数据库教程

图形图像教程

大数据教程

开发工具教程

计算机教程