将 CKServerChangeToken 保存到核心数据

Save CKServerChangeToken to Core Data

我使用 CloudKit 和 Core Data Swift 来同步我的数据。因此,我创建了订阅,并在收到通知后从云端获取新数据以更新我的核心数据。这是处理此更新的推荐方法。为了获取数据更改,我可以插入一个 CKServerChangeToken 来获取新内容。我的问题是如何将令牌保存到核心数据以供以后获取请求?对于 CKRecords,有一种方法可以仅保存元数据,但对于 CKServerChangeToken 则没有这样的方法。有人有想法吗?

此致,Jannik

CKServerChangeToken是一个不透明的数据对象,它继承自NSObject并符合NSCopying协议,这意味着您可以使用NSKeyedArchiver和NSKeyedUnarchiver将token转换为(NS)Data对象。

然后可以将 (NS)Data 对象存储到任何 NSManagedObject 上的 属性 中。或者,您可以将此数据存储在 (NS)UserDefaults 中。这是实现此目的的一种方法,作为对 UserDefaults 的 swift 扩展:

import Foundation
import CloudKit

public extension UserDefaults {

    public var serverChangeToken: CKServerChangeToken? {
        get {
            guard let data = self.value(forKey: "ChangeToken") as? Data else {
                return nil
            }

            guard let token = NSKeyedUnarchiver.unarchiveObject(with: data) as? CKServerChangeToken else {
                return nil
            }

            return token
        }
        set {
            if let token = newValue {
                let data = NSKeyedArchiver.archivedData(withRootObject: token)
                self.set(data, forKey: "ChangeToken")
            } else {
                self.removeObject(forKey: "ChangeToken")
            }
        }
    }
}

使用此扩展程序,您可以 get/set 来自 (NS)UserDefaults 的 CKServerChangeToken:

let changeToken = UserDefaults.standard.serverChangeToken
UserDefaults.standard.serverChangeToken = `newToken`

如前所述,NSKeyedUn/Archiver 调用在 iOS 12 中已弃用。这是一个更新的示例。

import Foundation
import CloudKit

public extension UserDefaults {

    public var serverChangeToken: CKServerChangeToken? {
        get {
            guard let data = self.value(forKey: "ChangeToken") as? Data else {
                return nil
            }

            let token: CKServerChangeToken?
            do {
                token = try NSKeyedUnarchiver.unarchivedObject(ofClass: CKServerChangeToken.self, from: data)
            } catch {
                token = nil
            }

            return token
        }
        set {
            if let token = newValue {
                do {
                    let data = try NSKeyedArchiver.archivedData(withRootObject: token, requiringSecureCoding: false)
                self.set(data, forKey: "ChangeToken")
                } catch {
                    // handle error
                }
            } else {
                self.removeObject(forKey: "ChangeToken")
            }
        }
    }
}