Często pomijanym problemem jest to, że ciągi znaków w formacie ISO 8601 mogą mieć milisekundy, a mogą nie.
Innymi słowy, zarówno „2016-12-31T23: 59: 59.9999999”, jak i „2016-12-01T00: 00: 00” są prawidłowe, ale jeśli używasz programu formatującego datę wpisywanego statycznie, jeden z nich nie zostanie przeanalizowany .
Począwszy od iOS 10 , powinieneś używać ISO8601DateFormatter
tego, który obsługuje wszystkie odmiany ciągów daty ISO 8601. Zobacz przykład poniżej:
let date = Date()
var string: String
let formatter = ISO8601DateFormatter()
string = formatter.string(from: date)
let GMT = TimeZone(abbreviation: "GMT")
let options: ISO8601DateFormatOptions = [.withInternetDateTime, .withDashSeparatorInDate, .withColonSeparatorInTime, .withTimeZone]
string = ISO8601DateFormatter.string(from: date, timeZone: GMT, formatOptions: options)
W przypadku systemu iOS 9 i starszych użyj następującego podejścia z wieloma elementami formatującymi dane.
Nie znalazłem odpowiedzi, która obejmowałaby oba przypadki i ukazywałaby tę subtelną różnicę. Oto rozwiązanie, które to rozwiązuje:
extension DateFormatter {
static let iso8601DateFormatter: DateFormatter = {
let enUSPOSIXLocale = Locale(identifier: "en_US_POSIX")
let iso8601DateFormatter = DateFormatter()
iso8601DateFormatter.locale = enUSPOSIXLocale
iso8601DateFormatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss.SSS'Z'"
iso8601DateFormatter.timeZone = TimeZone(secondsFromGMT: 0)
return iso8601DateFormatter
}()
static let iso8601WithoutMillisecondsDateFormatter: DateFormatter = {
let enUSPOSIXLocale = Locale(identifier: "en_US_POSIX")
let iso8601DateFormatter = DateFormatter()
iso8601DateFormatter.locale = enUSPOSIXLocale
iso8601DateFormatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss'Z'"
iso8601DateFormatter.timeZone = TimeZone(secondsFromGMT: 0)
return iso8601DateFormatter
}()
static func date(fromISO8601String string: String) -> Date? {
if let dateWithMilliseconds = iso8601DateFormatter.date(from: string) {
return dateWithMilliseconds
}
if let dateWithoutMilliseconds = iso8601WithoutMillisecondsDateFormatter.date(from: string) {
return dateWithoutMilliseconds
}
return nil
}
}
Stosowanie:
let dateToString = "2016-12-31T23:59:59.9999999"
let dateTo = DateFormatter.date(fromISO8601String: dateToString)
// dateTo: 2016-12-31 23:59:59 +0000
let dateFromString = "2016-12-01T00:00:00"
let dateFrom = DateFormatter.date(fromISO8601String: dateFromString)
// dateFrom: 2016-12-01 00:00:00 +0000
Polecam też sprawdzić artykułem Apple o formatach daty.