令人沮丧的是,我认为这是“正常”行为,尤其是对于具有大量要同步的关系的大型数据库 - 无法查看进度(向用户显示),也无法加快进度。
NSPersistentCloudKitContainer 似乎将每个关系视为单独的CKRecord,同步仍然受到相同限制(即一次不超过 400 个“请求”),您会经常看到这些 .limitExceeded 错误在控制台中,但几乎没有其他信息(即是否/何时重试??)。
我发现这导致数据库需要 天 天才能完全同步,同时数据看起来混乱且不完整。我有 数千个 多对多关系,当用户从外部文件恢复他们的数据库时,所有这些 CKRecords 都需要重新创建。
我在这里主要担心的是无法查询NSPersistentCloudKitContainer 是否有待处理的请求,还有多少尚未同步等,因此您可以将其转发给 UI 中的用户,以便他们不会一直认为它“失败”而删除和恢复。
当你关闭同步时本地数据被删除的一种方法 - 并且可能在你重新打开它时不必让它全部“重新同步” - 是在它关闭时使用NSPersistentContainer,然后使用NSPersistentCloudKitContainer 开启时。
NSPersistentCloudKitContainer 是NSPersistentContainer 的子类。
我目前正在我的自定义 PersistenceService 类中的应用程序中执行此操作:
static var useCloudSync = UserDefaults.standard.bool(forKey: "useCloudSync")
static var persistentContainer: NSPersistentContainer = {
let container: NSPersistentContainer?
if useCloudSync {
container = NSPersistentCloudKitContainer(name: "MyApp")
} else {
container = NSPersistentContainer(name: "MyApp")
let description = container!.persistentStoreDescriptions.first
description?.setOption(true as NSNumber,
forKey: NSPersistentHistoryTrackingKey)
}
container!.loadPersistentStores(completionHandler: { (storeDescription, error) in
if let error = error as NSError? {
fatalError("Unresolved error \(error), \(error.userInfo)")
}
})
return container!
}()
这至少会导致在您的应用中关闭 iCloud 时本地数据不会受到影响,并且在重新打开时不需要重新同步所有内容。
我想你也可以在NSPersistentCloudKitContainer删除所有本地数据之前查询iOS,看看用户是否在系统设置中关闭了iCloud,并在两者之间切换。
编辑:添加了NSPersistentHistoryTrackingKey,因为没有它,从NSPersistentCloudKitContainer 切换回NSPersistentContainer 失败。
它在我的应用程序中正常工作。当用户在我的应用程序中重新启用 iCloud 同步(并从 NSPersistentContainer 切换到 NSPersistentCloudKitContainer )时,它会同步自上次同步以来添加/更改的任何内容,这正是我想要的!
编辑 2:这是上述方法的更好实现
基本上,无论用户是否同步到 iCloud,只需将容器上的 .options 更改为使用 NSPersistentCloudKitContainerOptions(containerIdentifier:) 或 nil。似乎根本不需要在 NSPersistentCloudKitContainer 和 NSPersistentContainer 之间切换。
static var synciCloudData = {
return defaults.bool(forKey: Settings.synciCloudData.key)
}()
static var persistentContainer: NSPersistentContainer = {
let container = NSPersistentCloudKitContainer(name: "AppName")
guard let description = container.persistentStoreDescriptions.first else {
fatalError("Could not retrieve a persistent store description.")
}
description.setOption(true as NSNumber, forKey: NSPersistentHistoryTrackingKey)
if synciCloudData {
let cloudKitContainerIdentifier = "iCloud.com.yourID.AppName"
let options = NSPersistentCloudKitContainerOptions(containerIdentifier: cloudKitContainerIdentifier)
description.cloudKitContainerOptions = options
} else {
description.cloudKitContainerOptions = nil
}
container.loadPersistentStores(completionHandler: { (storeDescription, error) in
if let error = error as NSError? {
fatalError("Unresolved error \(error), \(error.userInfo)")
}
})
return container
}()
最后,您可以看到 iCloud 同步的状态,尽管很粗略(即,您无法看到是否有任何东西“同步”,只有同步处于挂起/正在进行以及是否它成功或失败。话虽如此,这足以满足我在我的应用程序中的用例。
查看用户 ggruen 在帖子底部的回复:
NSPersistentCloudKitContainer: How to check if data is synced to CloudKit