首页 > 解决方案 > Swift 中的绝对 UTC 偏移量

问题描述

我的任务是在 DST 和夏季时间创建一个包含 UTC 偏移量的字符串(例如:UTC+1UTC+2或者UTC+1UTC+1如果某个地区没有 DST)。我的函数如下所示:

extension TimeZone {
    public func utcOffset(for date: Date = Date()) -> String {
        var currentTimeOffest = self.secondsFromGMT(for: date)
        if isDaylightSavingTime() {
            currentTimeOffest -= Int(daylightSavingTimeOffset(for: date))
        }
        let currentInHours = Int(currentTimeOffest / 3_600)
        let hoursSymbol: String = currentInHours > 0 ? "+" : ""

        let daylightOffset = TimeInterval(currentTimeOffest) + self.daylightSavingTimeOffset(for: date)
        let daylightInHours = Int(daylightOffset / 3_600)
        let daylightSymbol: String = daylightInHours > 0 ? "+" : ""

        return "UTC\(hoursSymbol)\(currentInHours)UTC\(daylightSymbol)\(daylightInHours)"
    }
}

它运行良好,我已经为它编写了测试。一切都很好,但是在多个国家/地区最近的 DST 更改之后,测试开始失败,即使我通过了特定日期来计算以下偏移量:

    func testUtcOffset() {
        let date: Date = Date(timeIntervalSince1970: 1_557_482_400) //May 10, 2019 10:00:00 AM

        let warsaw = TimeZone.init(identifier: "Europe/Warsaw")! //eastern hemisphere, with DST
        XCTAssertEqual(warsaw.utcOffset(for: date), "UTC+2UTC+3")

        let shanghai = TimeZone.init(identifier: "Asia/Shanghai")! //eastern hemisphere, without DST
        XCTAssertEqual(shanghai.utcOffset(for: date), "UTC+8UTC+8")

        let barbados = TimeZone.init(identifier: "America/Barbados")! //western hemisphere, without DST
        XCTAssertEqual(barbados.utcOffset(for: date), "UTC-4UTC-4")

        let bermuda = TimeZone.init(identifier: "Atlantic/Bermuda")! //western hemisphere, with DST
        XCTAssertEqual(bermuda.utcOffset(for: date), "UTC-4UTC-3")

        let gmt = TimeZone.init(identifier: "GMT")! //GMT, without DST
        XCTAssertEqual(gmt.utcOffset(for: date), "UTC0UTC0")

        let lisbon = TimeZone.init(identifier: "Europe/Lisbon")! //GMT, with DST
        XCTAssertEqual(lisbon.utcOffset(for: date), "UTC+1UTC+2")
    }

2 周前,warsawlisbon时区开始失败,今天bermuda。有什么想法可能是错的吗?

标签: swifttimezoneutctimezone-offset

解决方案


一些东西:

  • 在您的测试中,华沙和里斯本的偏移量是一个小时。华沙标准时间为 UTC+1,夏令时为 UTC+2。里斯本在标准时间是 UTC+0,在白天是 UTC+1。

  • 从您的评论来看,您似乎正在寻找标准偏移量和日光偏移量。但是,标准偏移量不一定与当前偏移量相同。当前偏移量可能包括夏令时,也可能不包括。

  • 根据这些文档,如果一个有效,该secondsFromGMT函数会返回包括日光调整在内的差异。因此,您不应该自己为此进行调整。

  • daylightSavingTimeOffset在夏令时不适用的日期向函数询问偏移量似乎没有意义。secondsFromGMT仅使用当前年份的两个不同日期可能会获得更好的结果。一种常见的方法是获取 1 月 1 日和 7 月 1 日的偏移量。哪个较小的是标准时间,另一个是白天时间。请记住,如果不使用 DST,它们可能是相同的,并且它们将在北半球和南半球时区之间倒转。

    • 即使采用上述方法,这种算法也忽略了很多时区的复杂性。考虑到某些时区在其历史的不同时间点更改了标准时间。这样的算法可能会将其误认为是夏令时更改。
  • 需要注意的一点:一旦生成了字符串,例如"UTC+1UTC+2",您将如何单独从外部 API 知道要应用哪组夏令时规则?由于夏令时在世界不同地区的不同日期和时间开始和停止,因此在解释偏移时可能会使用错误的日期。


推荐阅读