Java没有关于所有IANA时区的信息
我正在尝试将来自前端的值映射到ZoneId类,如下所示:
Optional.ofNullable(timeZone).map(ZoneId::of).orElse(null) 对于大多数时区,它工作正常,但是,对于某些值,Java抛出异常: java.time.zone.ZoneRulesException: Unknown time-zone ID: America/Punta_Arenas 但是,根据IANA,它是一个有效的时区:
我在考虑为这些时区使用偏移量(只是硬编码值),但我想应该有更方便的方法来解决这个问题.有没有办法可以处理Java? 其他不受支持的时区: > America / Punta_Arenas 我的Java版本:“1.8.0_121”Java(TM)SE运行时环境(版本1.8.0_121-b13)Java HotSpot(TM)64位服务器VM(版本25.121-b13,混合模式) 解决方法
我已经使用Java 1.8.0_121进行了测试,并且一些区域确实缺失了.
最明显的解决方法是更新Java的版本 – 在Java 1.8.0_131中,上面的所有区域都可用 – 除了3个字母的名称(EST,HST等),更多内容如下. 但我知道生产环境中的更新并不像我们想的那么容易(也不快).在这种情况下,您可以使用TZUpdater tool,它可以在不更改Java版本的情况下更新JDK的时区数据. 唯一的细节是ZoneId不适用于3个字母的缩写(EST,HST等).那是因为这些名字是ambiguous and not standard. 但是,如果要使用它们,则可以使用自定义ID的映射. ZoneId带有内置地图: ZoneId.of("EST",ZoneId.SHORT_IDS); 问题在于choices used in the 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 当然,3个字母名称的唯一例外是GMT和UTC,但在这种情况下,最好只使用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; } } 创建ZoneRules是最复杂的部分. 一种方法是获取最新的IANA files并阅读它们.您可以查看JDK source code以了解它是如何创建ZoneRules的(尽管我不确定JDK中的文件是否与IANA的文件格式完全相同). 无论如何,this link解释了如何阅读IANA的文件.然后,您可以查看 然后你需要注册提供者: ZoneRulesProvider.registerProvider(new MissingZonesProvider()); 现在新区域将可用: ZoneId.of("America/Punta_Arenas"); ZoneId.of("Europe/Saratov"); ... and any other you added in the MissingZonesProvider class 有其他方法可以使用提供程序(而不是注册),check the javadoc以获取更多详细信息.在同一个javadoc中还有关于如何正确实现区域规则提供程序的更多细节(我的上面的版本非常简单,可能它缺少一些细节,比如provideVersions的实现 – 它应该使用提供者的版本作为密钥,而不是我正在做的区域ID等). 当然,一旦更新Java版本,就必须丢弃此提供程序(因为您不能有2个提供程序创建具有相同ID的区域:如果新提供程序创建已存在的ID,则在您尝试时会引发异常注册它). (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |