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]
相关推荐
若水无华2 天前
fiddler 配置ios手机代理调试
ios·智能手机·fiddler
Aress"2 天前
【ios越狱包安装失败?uniapp导出ipa文件如何安装到苹果手机】苹果IOS直接安装IPA文件
ios·uni-app·ipa安装
Jouzzy2 天前
【iOS安全】Dopamine越狱 iPhone X iOS 16.6 (20G75) | 解决Jailbreak failed with error
安全·ios·iphone
瓜子三百克2 天前
采用sherpa-onnx 实现 ios语音唤起的调研
macos·ios·cocoa
左钦杨2 天前
IOS CSS3 right transformX 动画卡顿 回弹
前端·ios·css3
努力成为包租婆2 天前
SDK does not contain ‘libarclite‘ at the path
ios
安和昂3 天前
【iOS】Tagged Pointer
macos·ios·cocoa
I烟雨云渊T4 天前
iOS 阅后即焚功能的实现
macos·ios·cocoa
struggle20254 天前
适用于 iOS 的 开源Ultralytics YOLO:应用程序和 Swift 软件包,用于在您自己的 iOS 应用程序中运行 YOLO
yolo·ios·开源·app·swift
Unlimitedz4 天前
iOS视频编码详细步骤(视频编码器,基于 VideoToolbox,支持硬件编码 H264/H265)
ios·音视频