Search code examples
swiftnsdatensdateformatterdateformatterswiftdate

Swift: Precision with timeIntervalSince1970


When I convert Unix epoch time to milliseconds, it's getting rounded.

Code:

import Foundation

extension Date {
    public func getMilliSeconds()-> TimeInterval {
        return TimeInterval(self.timeIntervalSince1970)
    }
}

let epochTime: TimeInterval = 1597269862.9328
print("\(epochTime) Precise Epoch Time /// (notice the 8 is the last digit)")

let convertedTime = Date(timeIntervalSince1970: epochTime)
print("\(convertedTime) Time converted To Swift Date")

let df = DateFormatter()
df.dateFormat = "y-MM-dd H:m:ss.SSSS"
let withMilli = df.string(from: convertedTime)
print("\(withMilli) Time with milliseconds using Date Formatter /// (notice 8 is missing and this was rounded to .9330")

if let convertedBackToEpochTime = df.date(from: withMilli) {
    print("\(convertedBackToEpochTime.getMilliSeconds()) Time converted back to Unix epoch time")
}

Log:

1597269862.9328 Precise Epoch Time /// (notice the 8 is the last digit)
2020-08-12 22:04:22 +0000 Time converted To Swift Date
2020-08-12 15:4:22.9330 Time with milliseconds using Date Formatter /// (notice 8 is missing and this was rounded to .9330
1597269862.933 Time converted back to Unix epoch time

I started with 1597269862.9328 and ended with 1597269862.933. How can I get the date back to exactly what I started with: 1597269862.9328?


Solution

  • The issue there is that DateFormatter is limited to milliseconds. There is no way to display more than 3 fraction digits using it.

    extension Formatter {
        static let iso8601withFractionalSeconds: DateFormatter = {
            let formatter = DateFormatter()
            formatter.calendar = Calendar(identifier: .iso8601)
            formatter.locale = Locale(identifier: "en_US_POSIX")
            formatter.timeZone = TimeZone(secondsFromGMT: 0)
            formatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss.SSSSSSSSSXXXXX"
            return formatter
        }()
    }
    

    Formatter.iso8601withFractionalSeconds.string(from: Date())  // "2020-10-29T16:12:27.111000000Z"
    

    If you would like to get the initial value back you just need top get the timeIntervalSince1970 from your convertedTime date:

    let epochTime: TimeInterval = 1597269862.9328
    print("\(epochTime) Precise Epoch Time")
    
    let convertedTime = Date(timeIntervalSince1970: epochTime)
    print("\(convertedTime) Time converted To Swift Date")
    
    print(convertedTime.timeIntervalSince1970)   // "1597269862.9328\n"
    

    Note that Swift Date is stored as the time interval since reference date. If you would like to preserve the date accuracy you should use timeIntervalSinceReferenceDate instead of timeIntervalSince1970 when archiving it. Check this post.