用异步序列优雅的监听 SwiftData 2.0 中历史追踪记录(History Trace)的变化

在这里插入图片描述

概述

WWDC 24 一声炮响为我们送来 Swift 6.0 的同时,也颇为“低调”的推出了 SwiftData 2.0。在新版本的 SwiftData 中,苹果为其新增了多个激动人心的新特性,其中就包括历史记录追踪(History Trace)。

在这里插入图片描述

不过,历史记录追踪目前看起来似乎有些“白璧微瑕”,略微让人有些不爽。在这里就让我们看看如何利用 Swift 结构化并发中的异步序列(AsyncSequence)来“补苴罅漏”吧。

相信通过本篇的学习,小伙伴们在精进 Swift 异步序列技艺的同时又能了然 SwiftData 2.0 的新“脾性”,何乐而不为呢?

闲言少叙,让我们马上开始吧!Let‘s go!!!😉


1. SwiftData 2.0 中的历史记录追踪

历史记录追踪(History Trace)是 SwiftData 2.0 中新推出的一种查询 SwiftData 数据库内容变化的机制。

History Trace “降生”的意义在于:利用它我们现在可以观察到不同模型上下文、不同进程以及系统不同组件对数据库内容的更改行为了。
在这里插入图片描述
在这里插入图片描述

举个例子:比如在 WatchOS 系统中包含共享同一个数据库(通过 App Groups)的 App 和 Widget。当 Widget 添加了一条记录后,我们的 App 如何能够知晓呢?

一种方法是在 App 进入前台时(active)被动读取数据库来发现变化。不过,更好的方法是让数据库自己主动告诉我们:底层数据发生了改变,需要秃头码农们的及时处理。

这可以通过在界面中监听 NSPersistentStoreRemoteChange 消息来实现:

.onReceive(NotificationCenter.default.publisher(for: .NSPersistentStoreRemoteChange).receive(on: DispatchQueue.main)) {
    _ in
    NSLog("数据库发生了变动!")
}

在得知数据库发生变化之后,我们随即就可以利用 History Trace 来“恣意”读取具体的历史 Change 记录了:

private func handleChangeInMainContext() {
   
    let mainContext = modelContext
    var historyDesc = HistoryDescriptor<DefaultHistoryTransaction>()
    historyDesc.predicate = #Predicate {
    trans in
        trans.author == "Widgets"
    }
    
    let transactions = try! mainContext.fetchHistory(historyDesc)
    for trans in transactions {
   
        for change in trans.changes {
   
            // 具体处理实现从略...
        }
    }
}

如上代码所示,当监听到底层数据库发生变动时我们可以调用 handleChangeInMainContext() 方法来查询所有实际变更的记录。从中我们还可以发现,我们利用了 #Predicate 宏来进行结果过滤从而只关注小组件(Widgets)引起的改变。


更多 SwiftData 2.0 中历史记录追踪和“墓碑”机制的详细介绍,请小伙伴们移步如下链接观赏系列文章(共 6 篇):


2. 一个小小的美中不足…

不知小伙伴们发现了没有,虽说利用 NSPersistentStoreRemoteChange 可以圆满的监听到 SwiftData 数据库的改变,但这种方式感觉把“监听”和“处理”操作隔离开了,无法从逻辑上体现出 Swift 语言的简洁和优雅。

参考苹果对于监听设备位置坐标改变实现的升级,我们希望在 SwiftData 2.0 的 History Trace 里也能用类似下面的代码来“抽丁拔楔”:

for await change in modelContext.persistentStoreChanges {
   
    // 对数据库中的改变进行处理...
}

看到这么熟悉且散发着 Swifty 范儿的“美味”代码,小伙伴们想必都会有一个似曾相识的“身影”映入脑海。别犹豫,大声说出来!它就是:异步序列

3. 异步序列的“将伯之助”

异步序列是 Swift 5.5+ 中跟随结构化异步模型推出的一种数据类型。系统内置框架本身就包含了海量异步序列,我们也可以遵守 AsyncSequence 协议来实现自己的异步序列。

在这里插入图片描述

更多关于异步序列的进一步精彩介绍,请小伙伴们移步如下链接观赏:

从之前的代码可以发现,我们对于历史记录的查询是在模型上下文对象上进行的。所以我们可以进一步扩展 ModelContext 类型来实现我们对应的异步序列:

extension ModelContext {
   
    var historyChanges: any AsyncSequence<(
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

大熊猫侯佩

赏点钱让我买杯可乐好吗 ;)

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值