我尝试将来自前端的值映射到ZoneId
类,如下所示:
Optional.ofNullable(timeZone).map(ZoneId::of).orElse(null)
对于大多数时区,它工作正常,但是,对于某些值,Java抛出异常:
java.time.zone.ZoneRulesException: Unknown time-zone ID: America/Punta_Arenas
但是,根据IANA,这是一个有效的时区: https://www.iana.org/time-zones
Zone America / Punta_Arenas -4:43:40 - LMT 1890
我在考虑为这些时区使用偏移量(只是硬编码值),但我想应该有更方便的方法来解决这个问题。有没有办法可以处理Java?
不受支持的其他时区:
我的Java版本:" 1.8.0_121" Java(TM)SE运行时环境(版本1.8.0_121-b13)Java HotSpot(TM)64位服务器VM(版本25.121-b13,混合模式)
答案 0 :(得分:8)
我已经使用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 SHORT_IDS
map - 就像任何其他选择一样 - 是任意的,甚至是有争议的。如果要为每个缩写使用不同的区域,只需创建自己的地图:
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的文件。然后,您可以查看ZoneRules
javadoc以了解如何将IANA信息映射到Java类。在this answer中,我创建了一个非常简单的ZoneRules
,只有2个转换规则,因此您可以基本了解如何执行此操作。
然后您需要注册提供者:
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
的实现 - 它应该使用提供程序的版本作为key,而不是我正在做的区域ID等。)
当然,一旦更新Java版本,就必须丢弃此提供程序(因为您不能有2个提供程序创建具有相同ID的区域:如果新提供程序创建已存在的ID,则会引发异常当你尝试注册时)。