Swift - 从ISO8601日期字符串中检索时区

时间:2018-05-17 06:12:53

标签: swift

我以这种格式将日期保存在数据库中 - " yyyy-MM-dd' H&H:mm:ssZ"。例如," 2018-05-17T11:15:00 + 0330"。时区变化,它在用户的本地时区中。

我希望检索并显示日期,例如" 2018年5月17日11.15AM",将日期/时间保存在与保存时相同的时区。我怎么能这样做?

我试过了。

let dateString = "2018-05-17T11:15:00+0330"
let formatter = DateFormatter()
formatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ssZ"
//formatter.timeZone = ???
print(formatter.date(from: dateString))

但我在本地时区获取Date对象并将其转换为日期字符串也会在本地时区显示日期。

在Swift中有这样的https://stackoverflow.com/a/46216251/1373592吗?

感谢。

2 个答案:

答案 0 :(得分:3)

你真正想做的是忽略时区。

您可以从日期字符串中删除最后5个字符,以一种格式解析它,并以另一种格式对其进行格式化:

let dateString = "2018-05-17T11:15:00+0330"
let formatter = DateFormatter()
formatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss"
let date = formatter.date(from: String(dateString.dropLast(5)))! // note the stripping off here
formatter.dateFormat = "MMMM d, yyyy hh.mma"
print(formatter.string(from: date))

请注意,您无法将此日期与其他日期进行比较,因为您已将其时区删除。

此外,SwiftDate是与JodaTime类似的库。

答案 1 :(得分:0)

这是TimeZone的一点扩展我刚刚掀起。您可以使用它从ISO8601日期字符串创建TimeZone。它将以下列格式处理字符串中的时区:

  • Z(UTC时间)
  • +/- HH
  • +/- HHMM
  • +/- HH:MM

这是扩展名:

extension TimeZone {
    init?(iso8601: String) {
        let tz = iso8601.dropFirst(19) // remove yyyy-MM-ddTHH:mm:ss part
        if tz == "Z" {
            self.init(secondsFromGMT: 0)
        } else if tz.count == 3 { // assume +/-HH
            if let hour = Int(tz) {
                self.init(secondsFromGMT: hour * 3600)
                return
            }
        } else if tz.count == 5 { // assume +/-HHMM
            if let hour = Int(tz.dropLast(2)), let min = Int(tz.dropFirst(3)) {
                self.init(secondsFromGMT: (hour * 60 + min) * 60)
                return
            }
        } else if tz.count == 6 { // assime +/-HH:MM
            let parts = tz.components(separatedBy: ":")
            if parts.count == 2 {
                if let hour = Int(parts[0]), let min = Int(parts[1]) {
                    self.init(secondsFromGMT: (hour * 60 + min) * 60)
                    return
                }
            }
        }

        return nil
    }
}

测试:

print(TimeZone(iso8601: "2018-05-17T11:15:00+0330"))

您可以将此解析和格式组合在一起,以便最终结果位于原始时区。

let dateString = "2018-05-17T11:15:00+0330"
let formatter = DateFormatter()
formatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ssZ"
formatter.locale = Locale(identifier: "en_US_POSIX")
if let date = formatter.date(from: dateString), let tz = TimeZone(iso8601: dateString) {
    let newFmt = DateFormatter()
    newFmt.dateStyle = .medium
    newFmt.timeStyle = .short
    newFmt.timeZone = tz
    let newString = newFmt.string(from: date)
    print(newString)
}