SwiftData数据同步到iCloud
SwiftData数据同步到iCloud

SwiftData数据同步到iCloud

SwiftData数据同步iCloud需要先配置iCloud容器。

配置iCloud容器

1、打开 Xcode 项目设置

在 Xcode 的项目导航中,点击项目的根目录,然后选择目标 (target) 设置。

2、进入 Signing & Capabilities

在项目设置中,点击 Signing & Capabilities 选项卡。

3、添加 iCloud 能力

点击左上方的 + Capability 按钮,搜索并选择 iCloud。这会自动为项目添加 iCloud 支持。

双击列表中的iCloud选项,将iCloud添加到项目中。

4、配置 iCloud 容器

选择添加的 iCloud 项目后,你会看到多个选项。确保勾选 CloudKit。

如果你的应用需要使用特定的容器,可以点击 Edit 按钮并选择或添加新的 iCloud 容器。

如果勾选CloudKit后,iCloud如果没有设置默认容器,则需要手动添加容器。

提示内容:如果命名的容器不存在,Xcode 将创建一个新的容器,将其添加到 App ID,然后将新容器添加到应用程序的权利中。

这里应该使用带有“iCloud”的应用程序包 ID 前缀,例如:

iCloud.com.fangjunyu.hackingwithswift

点击“OK”按钮并创建对应的CloudKit容器。

刚开始Containers容器可能是红色,可以点击下面的刷新按钮同步,同步后颜色变为黑色。

5、新增Remote Notifications功能

再次点击Capability功能:

找到并添加Background Modes功能:

在Background Modes中勾选Remote notifications功能。

勾选该功能后,当iCloud中的数据发送变化时,应用程序就会收到通知,就可以实现本地同步。

不勾选该功能,Xcode可能会输出类似的提示:

BUG IN CLIENT OF CLOUDKIT: CloudKit push notifications require the 'remote-notification' background mode in your info plist.

告知要求CloudKit 推送通知需要信息列表中的“远程通知”后台模式。

查看CloudKit容器

在 Apple Developer 网站上可以查看创建的 CloudKit 容器,但这需要在 Xcode 中成功创建容器并同步后,才会出现在 Apple Developer 网站的容器列表中。

1、打开 Apple Developer 网站

访问 Apple Developer 网站 并登录开发者账户。

2、导航到 CloudKit 容器管理页面

点击顶部菜单中的账户。

在页面中,选择服务-CloudKit(英文)。

3、查看容器列表

在 CloudKit 控制台页面,点击CloudKit数据库。

这里的页面是英文,使用的是谷歌页面翻译,实际的名称为CloudKit Database。

容器名称(如 iCloud.com.fangjunyu.hackingwithswift)应该出现在列表中。

如果未看到容器,请确保在 Xcode 中正确添加了 iCloud 功能并指定了容器。

CloudKit要求

配置CloudKit容器后,需要了解CloudKit对模型有严格要求:

1、所有的属性必须有 默认值 或 可选类型。

2、所有关系也必须是 可选的。

检查创建的模型。如果之前的修复未完全满足这些要求(如遗漏了某些字段的默认值或未声明为可选类型),仍会导致 ModelContainer 初始化失败。

初始化失败截图:

会提示下面类似的报错:

Thread 1: Fatal error: 'try!' expression unexpectedly raised an error: SwiftData.SwiftDataError(_error: SwiftData.SwiftDataError._Error.loadIssueModelContainer)

参考模型

下面是提供默认值的属性,都需要添加默认值,如果是关系则需要为可选类型:

@Model
class User {
    var name: String = "Anonymous"
    var city: String = "Unknown"
    var joinDate: Date = Date.now
    @Relationship(deleteRule: .cascade) var jobs: [Job]? = [Job]()

    init(name: String, city: String, joinDate: Date, jobs: [Job]? = nil) {
        self.name = name
        self.city = city
        self.joinDate = joinDate
        self.jobs = jobs
    }
}

我在添加默认值和可选类型过程中,因为忘记给关系添加可选类型报错,一时间还找到问题根源,因此需要注意类似的问题:

[Job]? = [Job]()

配置iCloud容器

最后需要在入口文件的ModelContainer中配置iCloud容器。

import SwiftUI
import SwiftData
@main
struct pigletApp: App {
    @State private var modelConfigManager = ModelConfigManager()
    var body: some Scene {
        WindowGroup {
            ContentView()
        }
        .environment(modelConfigManager)
        .environmentObject(iapManager)
        .modelContainer(try! ModelContainer(for: PiggyBank.self,SavingsRecord.self,configurations: ModelConfiguration(
            "PrivateDatabaseContainer",
            cloudKitDatabase: .private("iCloud.com.fangjunyu.piglet")
        )))
    }
}

需要注意的是,Xcode项目一旦启用iCloud,modelContainer必须配置cloudKitDatabase,否则运行报错。

cloudKitDatabase使用的是 .private私有数据库,私有数据库后面的名称是前面创建的iCloud容器名称,名称不一致也会报错。

关于iCloud数据库类型的知识,可以进一步阅读《Apple CloudKitk框架》一文。

如果Xcode开启iCloud,但是不想要同步,可以设置cloudKitDatabase为.none。

cloudKitDatabase: .none

全部配置完成后,如果没有报错,应用在调试时,Xcode会正常输出SwiftData同步的相关信息。

最后,也可以在iCloud存储空间(设置-iCloud-管理账户储存空间)中进行验证。

总结

在Xcode中首先需要正确的配置iCloud容器,接着是Swift Data模型的所有字段都需要设置默认值,所有的关系都必须设置为可选类型,否则运行报错。

最后在ModelContainer容器加载时,配置cloudKitDatabase参数。

在同步的过程中可能存在很多问题,每一步都需要仔细的检查。

如果关系设置为可选类型后,原来的代码逻辑可能会改变,比如之前是直接校验关系数组是否为空。

if piggyBank.records.isEmpty { }

但是在修改records字段为可选后,原先的代码就会报错。因此,需要考虑解包:

let records = piggyBank.records ?? [] // 解包处理
if records.isEmpty { }

最后的实现效果为,每次处理SwiftData数据时,数据都会同步到iCloud。即使卸载应用,重新安装应用时,SwiftData仍然会从iCloud中正确读取相关数据。

补充知识

iCloud三种不同的数据存储方式

在 iCloud 中,Key-value storage、iCloud Documents 和 CloudKit 是三种不同的数据存储方式,分别适用于不同的应用场景:

1、Key-value storage

用途:适用于保存小量、简单的用户数据,例如设置、偏好和轻量级的应用状态。

特点

提供一个键-值对(key-value pairs)的存储模型,类似于 NSUserDefaults。

数据可以在多个设备之间同步。

由于存储量小且同步速度快,适合于用户设置和状态信息(如主题、音量设置等)的跨设备同步。

示例:在 iPhone 上保存的用户设置可以通过 iCloud 同步到 iPad。

2、iCloud Documents

用途:专门用于存储用户文件和文档,例如文本文件、图片、视频等。

特点

为用户的文件提供集中存储,并能在所有设备上访问。

支持应用数据的跨设备无缝同步,但用户可以直接访问这些文件,因此适合用户文件管理的需求。

Apple 提供了 UIDocument 类来帮助管理文件,处理文件的读写和保存过程。

示例:iCloud Drive 上的文档,如 Pages、Numbers 或用户上传的 PDF 文件等。

3、CloudKit

用途:提供一个云端数据库服务,适用于保存更复杂的结构化数据,如应用的核心数据、关系型数据等。

特点

支持公有和私有数据库,可以存储大规模数据。

适合复杂数据结构,如包含关系的用户数据、共享内容、或应用的核心数据存储。

开发者可以利用 CloudKit 的 API 创建、读取、更新和删除记录。支持用户间数据共享,且具备权限控制。

示例:社交类应用的用户动态信息、跨设备的内容共享(例如笔记和提醒)、游戏的多人数据等。

使用场景对比:

Key-value storage 适合轻量级配置和偏好设置。

iCloud Documents 用于文件存储和管理,适合用户生成的内容。

CloudKit 则适合需要数据库功能的应用,能存储结构化和关联性数据,支持数据同步和复杂的应用逻辑。

本地数据同步问题

在勾选 CloudKit 后,UserDefaults 和 URL 指向的 Documents 目录中的数据不会自动同步到 iCloud。CloudKit 和 iCloud 并不自动同步本地数据,需要特定的 API 或设置。以下是它们的存储行为与 iCloud 的关系:

1、UserDefaults

本地存储:默认情况下,UserDefaults 只在本地设备上存储数据,不会自动与 iCloud 同步。

iCloud Key-Value Storage:如果希望在 iCloud 中同步 UserDefaults 的部分数据,可以通过 NSUbiquitousKeyValueStore,它提供了键值存储(key-value storage)与 iCloud 的同步。这个功能适合简单的偏好设置和小数据量的同步。

使用 iCloud Key-Value Storage 时,需要在 Xcode 中为项目勾选 iCloud 的 Key-value storage 功能,否则数据不会同步到 iCloud。

如何使用

let iCloudStore = NSUbiquitousKeyValueStore.default

// 保存数据
iCloudStore.set("John Doe", forKey: "username")

// 读取数据
let username = iCloudStore.string(forKey: "username")
iCloudStore.synchronize()

需要注意,这种同步仅限于轻量级数据,数据量大的内容需要其他解决方案,比如 CloudKit。

2、Documents 目录中的文件(URL Documents Storage)

本地存储:通常存储在 Documents 或 Library 中的文件,默认也不会自动同步到 iCloud。

iCloud Drive 手动存储:如果你希望将某些文件同步到 iCloud,开发者需要主动将文件存储到 iCloud Drive 目录下(例如在 Documents 目录中创建一个 iCloud 目录并将文件写入)。

FileManager 的 url(forUbiquityContainerIdentifier:) 方法用于访问 iCloud Documents 功能的存储路径。将文件写入这个路径可以实现文件的 iCloud 同步,但这需要在 Xcode 中启用 iCloud Documents 功能

使用方式

let iCloudURL = FileManager.default.url(forUbiquityContainerIdentifier: nil)?.appendingPathComponent("Documents").appendingPathComponent("yourFileName")
// 将文件写入到 iCloudURL 指定的位置

NSUbiquitousKeyValueStore是什么?

     NSUbiquitousKeyValueStore 是 Apple 提供的 iCloud 功能,用于在轻量级应用数据(如用户偏好设置、配置或状态)在启用 iCloud 的 Apple 设备之间进行同步。它不是 SwiftUI 的特有功能,而是 Foundation 框架的一部分,可以在 Swift 和 Objective-C 中使用。

什么是 NSUbiquitousKeyValueStore?

     iCloud 的轻量级键值存储:主要用于存储小型数据,例如用户设置、应用配置等。每个应用的总存储空间有限(当前约 1 MB),适合频繁更新但体量小的数据。

     跨设备同步:存储的数据会自动通过 iCloud 同步到用户的其他 Apple 设备。只要用户在设备上登录同一个 Apple ID 并开启 iCloud,数据就会保持一致。

NSUbiquitousKeyValueStore 的工作方式

本地与 iCloud 数据的同步:数据存入 NSUbiquitousKeyValueStore 后,会自动同步到 iCloud,在其他设备上也能读取和使用。

手动同步:可以通过 synchronize() 方法手动触发数据同步,确保最新数据上传到 iCloud 或从 iCloud 下载更新。

基本用法

在 Swift 中使用时,我们首先要访问默认的 NSUbiquitousKeyValueStore 实例,然后存储或读取数据:

let iCloudStore = NSUbiquitousKeyValueStore.default

// 保存数据
iCloudStore.set("John Doe", forKey: "username")

// 读取数据
let username = iCloudStore.string(forKey: "username")

相关文章

Apple CloudKit框架:https://fangjunyu.com/2025/01/04/apple-cloudkit%e6%a1%86%e6%9e%b6/

如果您认为这篇文章给您带来了帮助,您可以在此通过支付宝或者微信打赏网站开放者。

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注