Java没有关于所有IANA时区的信息

dve*_*opp 6 java timezone timezone-offset java-time

我正在尝试将来自前端的值映射到ZoneId类,如下所示:

Optional.ofNullable(timeZone).map(ZoneId::of).orElse(null)
Run Code Online (Sandbox Code Playgroud)

对于大多数时区,它工作正常,但是,对于某些值,Java抛出异常:

java.time.zone.ZoneRulesException: Unknown time-zone ID: America/Punta_Arenas
Run Code Online (Sandbox Code Playgroud)

但是,根据IANA,它是一个有效的时区:https: //www.iana.org/time-zones

Zone America/Punta_Arenas -4:43:40 - LMT 1890

我在考虑为这些时区使用偏移量(只是硬编码值),但我想应该有更方便的方法来解决这个问题.有没有办法可以处理Java?

其他不受支持的时区:

  • 美国/ Punta_Arenas
  • 亚洲/阿特劳
  • 亚洲/法马古斯塔
  • 亚洲/仰光
  • 美东时间
  • 欧洲/萨拉托夫
  • HST
  • MST

我的Java版本:"1.8.0_121"Java(TM)SE运行时环境(版本1.8.0_121-b13)Java HotSpot(TM)64位服务器VM(版本25.121-b13,混合模式)

小智 10

我已经使用Java 1.8.0_121进行了测试,并且一些区域确实缺失了.

要解决这个问题,最明显的办法是更新Java的版本-在Java中1.8.0_131所有上述区域可供选择-除了3个字母的名字(EST,HST,等),低于更多.

但我知道生产环境中的更新并不像我们想的那么容易(也不快).在这种情况下,您可以使用TZUpdater工具,该工具可以在不更改Java版本的情况下更新JDK的时区数据.


唯一的细节是ZoneId不能使用3个字母的缩写(EST,HST依此类推).那是因为这些名字含糊不清而且不标准.

但是,如果要使用它们,则可以使用自定义ID的映射.ZoneId附带内置地图:

ZoneId.of("EST", ZoneId.SHORT_IDS);
Run Code Online (Sandbox Code Playgroud)

问题在于地图使用SHORT_IDS选择 - 像任何其他选择一样 - 是任意的,甚至是有争议的.如果要为每个缩写使用不同的区域,只需创建自己的地图:

Map<String, String> map = new HashMap<>();
map.put("EST", "America/New_York");
... put how many names you want
System.out.println(ZoneId.of("EST", map)); // creates America/New_York
Run Code Online (Sandbox Code Playgroud)

当然,3个字母名称的唯一例外是GMTUTC,但在这种情况下,最好只使用ZoneOffset.UTC常量.


如果您无法更新Java版本或运行TZUpdater工具,那么还有另一种(更难的)替代方案.

您可以扩展java.time.zone.ZoneRulesProvider该类并创建一个可以创建缺失ID的提供程序.像这样的东西:

public class MissingZonesProvider extends ZoneRulesProvider {

    private Set<String> missingIds = new HashSet<>();

    public MissingZonesProvider() {
        missingIds.add("America/Punta_Arenas");
        missingIds.add("Europe/Saratov");
        // add all others
    }

    @Override
    protected Set<String> provideZoneIds() {
        return this.missingIds;
    }

    @Override
    protected ZoneRules provideRules(String zoneId, boolean forCaching) {
        ZoneRules rules = null;
        if ("America/Punta_Arenas".equals(zoneId)) {
            rules = // create rules for America/Punta_Arenas
        }
        if ("Europe/Saratov".equals(zoneId)) {
            rules = // create rules for Europe/Saratov
        }
        // and so on

        return rules;
    }

    // returns a map with the ZoneRules, check javadoc for more details
    @Override
    protected NavigableMap<String, ZoneRules> provideVersions(String zoneId) {
        TreeMap<String, ZoneRules> map = new TreeMap<>();
        ZoneRules rules = provideRules(zoneId, false);
        if (rules != null) {
            map.put(zoneId, rules);
        }
        return map;
    }
}
Run Code Online (Sandbox Code Playgroud)

创造ZoneRules最复杂的部分.

一种方法是获取最新的IANA文件并阅读它们.您可以查看JDK源代码以了解它ZoneRules是如何创建的(尽管我不确定JDK中的文件是否与IANA文件的格式完全相同).

无论如何,此链接解释了如何阅读IANA的文件.然后,您可以查看ZoneRulesjavadoc以了解如何将IANA信息映射到Java类.在这个答案中,我创建了一个非常简单ZoneRules的只有2个转换规则,因此您可以基本了解如何执行此操作.

然后你需要注册提供者:

ZoneRulesProvider.registerProvider(new MissingZonesProvider());
Run Code Online (Sandbox Code Playgroud)

现在新区域将可用:

ZoneId.of("America/Punta_Arenas");
ZoneId.of("Europe/Saratov");
... and any other you added in the MissingZonesProvider class
Run Code Online (Sandbox Code Playgroud)

还有其他方法可以使用提供程序(而不是注册),请查看javadoc以获取更多详细信息.在同一个javadoc中还有关于如何正确实现区域规则提供程序的更多细节(我的上面版本非常简单,可能它缺少一些细节,比如实现provideVersions- 它应该使用提供程序的版本作为密钥,而不是区域ID正如我正在做的那样).

当然,一旦更新Java版本,就必须丢弃此提供程序(因为您不能有2个提供程序创建具有相同ID的区域:如果新提供程序创建已存在的ID,则在您尝试时会引发异常注册它).