SwiftData 学习笔记:relationship

前言

在前几篇文章中,我们介绍了 SwiftData 的简单使用和增删改查操作,以及数据的排序过滤和 Container、context 的概念介绍。

本篇文章主要来讲解一下 SwiftData 中的 relationship。

自动推断

SwiftData 的 relationship 是可以通过你定义的模型结构体自动推断出来的。也可以通过 @Relationship 宏来显式声明。

一般来说,只有当你不想使用默认配置的时候,才需要显式声明。通常情况下,你是用不到 @Relationship 的。

比如下面的这个例子:

swift 复制代码
@Model
class School {
    var name: String
    var students: [Student]

    init(name: String, students: [Student]) {
        self.name = name
        self.students = students
    }
}

@Model
class Student {
    var name: String
    var school: School

    init(name: String, school: School) {
        self.name = name
        self.school = school
    }
}

我们定义了一个 School 和一个 Student的类,并且,在 Student 中使用了 School 声明属性;在 School 中使用 Student 声明一个数组。从上述代码中,SwiftData 可以推断出:

  • 每个学校可以有很多学生
  • 每个学生只属于一所学校

但是,这两者并不是耦合在一起的,它们是分开的。如果我们创建了一个学生并设置了它的 school属性,SwiftData 不理解如何将该学生添加到该学校的 students 数组中,它不会自动推断出这种关系是双向的。

如果,我们将 Student 改成下面的代码:

swift 复制代码
@Model
class Student {
    var name: String
    var school: School?

    init(name: String, school: School?) {
        self.name = name
        self.school = school
    }
}

唯一的变化是我们将 school 声明为了可选类型,这意味着它的值可能为空。做出这一改动是基于安全的考虑:

  • 如果学生和学校之间存在关系,那么设置学生的 school 属性应该将其添加到学校的学生列表中或将其从学校的学生列表中删除
  • 同样,从学校的学生列表中添加或者删除学生,也应该同步修改学生的 school 属性
  • 那么,如果你将一名学生从一所学校中删除,而不将其添加到另一所学校,会发生什么情况呢?

当我们将 school 属性声明为非可选值的时候,即意味着上述情况是不可能发生的,学生必须归属于一所学校。如果学生的 school 属性没有被赋值,SwiftData 会触发 crash,因为我们将其置于无效状态。

另一方面,一旦我们将 school 属性设置为可选属性,这个危险就消失了:从 students 数组中删除一个学生只会将他们的 school 属性设置为 nil ,所以就没有了崩溃风险。

这里的规则很简单:如果可以安全地推断关系,SwiftData 就会去自动推断。

显式声明

很多时候,这还不够,我们可以在两个模型之一上使用 @Relationship宏创建显式关系,它显式地说明连接。例如,我们可以改变Student类,让它的school属性看起来像这样:

java 复制代码
@Relationship(inverse: \School.students) var school: School

这可以是可选的,也可以是非可选的------这里没有安全限制,因为你已经准确地告诉 SwiftData 你想要什么。

或者,我们可以更改 School 类,使其学生属性如下所示:

java 复制代码
@Relationship(inverse: \Student.school) var students: [Student]
相关推荐
游戏开发爱好者81 小时前
日常开发与测试的 App 测试方法、查看设备状态、实时日志、应用数据
android·ios·小程序·https·uni-app·iphone·webview
黑码哥2 小时前
ViewHolder设计模式深度剖析:iOS开发者掌握Android列表性能优化的实战指南
android·ios·性能优化·跨平台开发·viewholder
2501_915106323 小时前
app 上架过程,安装包准备、证书与描述文件管理、安装测试、上传
android·ios·小程序·https·uni-app·iphone·webview
2501_915106324 小时前
使用 Sniffmaster TCP 抓包和 Wireshark 网络分析
网络协议·tcp/ip·ios·小程序·uni-app·wireshark·iphone
熊猫钓鱼>_>4 小时前
移动端开发技术选型报告:三足鼎立时代的开发者指南(2026年2月)
android·人工智能·ios·app·鸿蒙·cpu·移动端
徐同保1 天前
通过ip访问nginx的服务时,被第一个server重定向了,通过设置default_server解决这个问题
ios·iphone
2501_915918411 天前
在 iOS 环境下查看 App 详细信息与文件目录
android·ios·小程序·https·uni-app·iphone·webview
2501_916007471 天前
没有 Mac 用户如何上架 App Store,IPA生成、证书与描述文件管理、跨平台上传
android·macos·ios·小程序·uni-app·iphone·webview
夏幻灵2 天前
HTTPS全面解析:原理、加密机制与证书体
ios·iphone
TheNextByte12 天前
如何在iPhone上恢复已删除的笔记的综合指南
笔记·ios·iphone