我正在尝试将来自前端的值映射到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?

其他不受支持的时区:

> America / Punta_Arenas
>亚洲/阿特劳
>亚洲/法马古斯塔
>亚洲/仰光
> EST
>欧洲/萨拉托夫
> HST
> MST
> ROC

我的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 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的实现 – 它应该使用提供者的版本作为密钥,而不是我正在做的区域ID等).

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

Java没有关于所有IANA时区的信息的更多相关文章

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. js中‘!.’是什么意思

  3. InnoDB 和 MyISAM 引擎恢复数据库,使用 .frm、.ibd文件恢复数据库

  4. Error: Cannot find module ‘node:util‘问题解决

    控制台 安装 Vue-Cli 最后一步出现 Error: Cannot find module 'node:util' 问题解决方案1.问题C:\Windows\System32>cnpm install -g @vue/cli@4.0.3internal/modules/cjs/loader.js:638 throw err; &nbs

  5. yarn的安装和使用(全网最详细)

    一、yarn的简介:Yarn是facebook发布的一款取代npm的包管理工具。二、yarn的特点:速度超快。Yarn 缓存了每个下载过的包,所以再次使用时无需重复下载。 同时利用并行下载以最大化资源利用率,因此安装速度更快。超级安全。在执行代码之前,Yarn 会通过算法校验每个安装包的完整性。超级可靠。使用详细、简洁的锁文件格式和明确的安装算法,Yarn 能够保证在不同系统上无差异的工作。三、y

  6. 前端环境 本机可切换node多版本 问题源头是node使用的高版本

    前言投降投降 重头再来 重装环境 也就分分钟的事 偏要折腾 这下好了1天了 还没折腾出来问题的源头是node 使用的高版本 方案那就用 本机可切换多版本最终问题是因为nodejs的版本太高,导致的node-sass不兼容问题,我的node是v16.14.0的版本,项目中用了"node-sass": "^4.7.2"版本,无法匹配当前的node版本根据文章的提

  7. 宝塔Linux的FTP连接不上的解决方法

    宝塔Linux的FTP连接不上的解决方法常见的几个可能,建议先排查。1.注意内网IP和外网IP2.检查ftp服务是否启动 (面板首页即可看到)3.检查防火墙20端口 ftp 21端口及被动端口39000 - 40000是否放行 (如是腾讯云/阿里云等还需检查安全组)4.是否主动/被动模式都不能连接5.新建一个用户看是否能连接6.修改ftp配置文件 将ForcePassiveIP前面的#去掉 将19

  8. 扩展element-ui el-upload组件,实现复制粘贴上传图片文件,带图片预览功能

  9. 微信小程序canvas实现水平、垂直居中效果

    这篇文章主要介绍了小程序中canvas实现水平、垂直居中效果,本文图文实例代码相结合给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下

  10. 使用HTML5做的导航条详细步骤

    这篇文章主要介绍了用HTML5做的导航条详细步骤,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

随机推荐

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. Mybatis分页插件PageHelper手写实现示例

    这篇文章主要为大家介绍了Mybatis分页插件PageHelper手写实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  4. (jsp/html)网页上嵌入播放器(常用播放器代码整理)

    网页上嵌入播放器,只要在HTML上添加以上代码就OK了,下面整理了一些常用的播放器代码,总有一款适合你,感兴趣的朋友可以参考下哈,希望对你有所帮助

  5. Java 阻塞队列BlockingQueue详解

    本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景,通过实例代码介绍了Java 阻塞队列BlockingQueue的相关知识,需要的朋友可以参考下

  6. Java异常Exception详细讲解

    异常就是不正常,比如当我们身体出现了异常我们会根据身体情况选择喝开水、吃药、看病、等 异常处理方法。 java异常处理机制是我们java语言使用异常处理机制为程序提供了错误处理的能力,程序出现的错误,程序可以安全的退出,以保证程序正常的运行等

  7. Java Bean 作用域及它的几种类型介绍

    这篇文章主要介绍了Java Bean作用域及它的几种类型介绍,Spring框架作为一个管理Bean的IoC容器,那么Bean自然是Spring中的重要资源了,那Bean的作用域又是什么,接下来我们一起进入文章详细学习吧

  8. 面试突击之跨域问题的解决方案详解

    跨域问题本质是浏览器的一种保护机制,它的初衷是为了保证用户的安全,防止恶意网站窃取数据。那怎么解决这个问题呢?接下来我们一起来看

  9. Mybatis-Plus接口BaseMapper与Services使用详解

    这篇文章主要为大家介绍了Mybatis-Plus接口BaseMapper与Services使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. mybatis-plus雪花算法增强idworker的实现

    今天聊聊在mybatis-plus中引入分布式ID生成框架idworker,进一步增强实现生成分布式唯一ID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部