因在之前自己写的一款 RSS 阅读器 应用中,有不少用户反馈说希望添加个 iCloud 同步订阅列表的功能,加上自己也对 iCloud 这一块的不熟悉,所以闲来研究下 iCloud 的内容。上述应用是支持 iPhone 和 iPad 的,两端同时使用时,没有同步功能的话,的确是一件很蛋痛的事情,这里先来分享下如果通过 iCloud 来同步应用的偏好设置。
配置环境
在项目中配置下 iCloud 的证书,现在 apple 相关的东西还是比较友好了,在 Xcode 中勾选相应配置就可以自动生成对应的证书,如下:
这里只用到了 Key-value storage,CloudKit 是可以不勾选的。
通过模拟器来调试 iCloud
在模拟器中调试 iCloud 是和真机类似的,都是在设置中登录 iCloud 的账号,但是在模拟器中,iCloud 是不是自动同步的,需要检查同步的时候需要手动执行,我们通过 Xcode 中的 Debug 功能来执行:
同步
-
保存 Key-value 到 iCloud
NSUbiquitousKeyValueStore.default().set(true, forKey: kOffLineRead)
在修改偏好设置的时候,把相应的值同步到 iCloud。
-
监听变化
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool { NotificationCenter.default.addObserver(self, selector: #selector(updateCloudItems(_ :)), name: NSUbiquitousKeyValueStore.didChangeExternallyNotification, object: NSUbiquitousKeyValueStore.default()) } func updateCloudItems(_ notification: Notification) { guard let userInfo = notification.userInfo else { return } guard let reason = userInfo[NSUbiquitousKeyValueStoreChangeReasonKey] as? Int else { return } if reason == NSUbiquitousKeyValueStoreServerChange || reason == NSUbiquitousKeyValueStoreInitialSyncChange { guard let changeKeys = userInfo[NSUbiquitousKeyValueStoreChangedKeysKey] as? [String] else { return } if changeKeys.contains(kOffLineRead) { if let value = NSUbiquitousKeyValueStore.default().object(forKey: kOffLineRead) as? Bool { UserDefaults.standard.set(value, forKey: kOffLineRead) UserDefaults.standard.synchronize() } } } }
因为这里的偏好设置应用了整个应用,所以在程序启动时就去监听 iCloud 的变化,如果只是某个页面的展示内容,可以考虑在性展示的时候再去监听。这里需要注意的是,iCloud 对 Key-value 的大小是有限制的,以前好像是2M,现在没有测试过,所以大的东西不要用 Key-value 来保存。