无法从此特定日期(1994-04-01)字符串转换为NSDate

时间:2016-12-14 15:47:07

标签: ios nsdateformatter

有些奇怪的事发生在我身上。 我无法转换此特定日期(1994-04-01)字符串。 任何人都可以检查这个并让我知道它是否在您的代码中重现?

重现步骤:

夫特: -

let dateString = "1994-04-01"
let dateFormatter = DateFormatter()
dateFormatter.dateFormat = "yyyy-MM-dd"
let dateFromString = dateFormatter.date(from: dateString)

对象: -

NSDateFormatter *formatter = [[NSDateFormatter alloc]init];
[formatter setDateFormat:@"yyyy-MM-dd"];
NSString *birthdayStr = @"1994-04-01";
NSDate *birthday = [formatter dateFromString:birthdayStr];

预期结果:

birthday = 1994-03-31 21:00:00 +0000

UTC

实际结果:

birthday = nil

版本:

Xcode ver: - 版本8.1(8B62)

OS X ver: - 10.12.1(16B2555)

你可以尝试任何不同的日期(1994-04-01),它会正常工作。

1 个答案:

答案 0 :(得分:7)

测试代码:

import Foundation

for timeZoneId in TimeZone.knownTimeZoneIdentifiers {
    let dateString = "1994-04-01"
    let dateFormatter = DateFormatter()
    dateFormatter.timeZone = TimeZone(identifier: timeZoneId)
    dateFormatter.dateFormat = "yyyy-MM-dd"
    if dateFormatter.date(from: dateString) == nil {
        print(timeZoneId)
    }
}

输出:

Asia/Amman
Asia/Damascus
Asia/Gaza
Asia/Hebron
Asia/Jerusalem

我总结您的系统时区设置为打印的时区之一。如果我在谷歌输入“耶路撒冷时区1994”,the first result告诉我,夏令时从1994年4月1日午夜开始。这意味着没有午夜。 1994年4月1日的第一个时刻实际上是该时区的凌晨1点。

默认情况下,DateFormatter使用午夜时段,而不会解析字符串中的时间。这使得它在字符串中的日期不存在午夜时失败。

解决方案是不使用午夜作为您的默认时间。中午是一个更安全的默认时间。因此,一种解决方案是在输入中包含时间并以以下格式解析它:

    let dateString = "1994-04-01 12:00:00"
    let dateFormatter = DateFormatter()
    dateFormatter.timeZone = TimeZone(identifier: timeZoneId)
    dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss"

另一种解决方案是为日期格式化程序提供某天中午的默认日期:

    // Reference date was 00:00:00 UTC on 1 January 2001. This is noon of the same day in UTC.
    dateFormatter.defaultDate = Date(timeIntervalSinceReferenceDate: 12*60*60)

如果您要在iOS或macOS上进行任何日期解析或操作,那么观看WWDC 2013 Session 227: Solutions to Common Date and Time Challenges 非常好主意。