在计算 属性 上对 Realm 结果集合进行排序?

Sorting Realm results collection on computed property?

我在我的 Swift iOS 应用程序中使用 Realm,并且我有一个模型,我正在尝试向其中添加计算的 属性。这是 class 的样子:

class Conversation: Object {

    @objc dynamic var conversationId: String = generateConversationId()
    @objc dynamic var createdTime = Date()
    @objc dynamic var title: String = ""
    let messages = List<Message>()

    var lastUpdated: Date {
        if let newestMessage = self.messages.sorted(byKeyPath: "timestamp", ascending: false).first {
            return newestMessage.timestamp
        } else {
            return Date(timeIntervalSince1970: 0)
        }
    }        
}

如您所见,此对象表示具有多个属性的对话,其中一个是属于该对话的消息列表。我添加的计算 属性 (lastUpdated) 应该只是 return messages 列表中最新消息的时间戳,或者当列表为空时,它return 默认日期值。

在我的一个视图控制器中,我正在创建一个结果集合来访问 Conversation 对象,如下所示:

var conversations: Results<Conversation> = realm.objects(Conversation.self)

这有效,我可以按预期在每个 Conversation 上访问 lastUpdated 属性。但是,我想让这个集合按 lastUpdated 属性 排序,所以我尝试像这样修改我的结果集合:

var conversations: Results<Conversation> = realm.objects(Conversation.self).sorted(byKeyPath: "lastUpdated", ascending: true)

当我这样做时,应用程序抛出以下异常:

Terminating app due to uncaught exception 'RLMException', reason: 'Cannot sort on key path 'lastUpdated': property 'Conversation.lastUpdated' does not exist.'

认为 这可能会发生,因为 lastUpdated 计算的 属性 没有持久保存在 Conversation 对象上(我会使用Realm Studio 可以快速验证这一点,但我的 Realm 是加密的)。我没有明确忽略这个 属性 但我认为是这种情况,因为 lastUpdated 没有像我所有其他属性一样用 @objc dynamic 装饰。我尝试添加它,擦除我的应用程序安装,重新编译并再次测试,但我仍然遇到相同的异常。

也许无法保留计算的 属性(因为,好吧,它是计算的!)但是是否有另一种方法可以指定我的对话结果集合应按此 属性 排序?

正如您所怀疑的,您无法持久化计算的 属性 并且 Realmsorted(byKeyPath:,ascending:) 方法仅适用于持久化属性。在保持自动更新 Results.

的同时,无法根据计算的 属性 对 Results 实例进行排序

但是,如果您不需要 Results 的自动更新特性,则有一个解决方法。您可以使用 Swift 的 sorted 方法,该方法将闭包作为其排序参数,这将导致类型为 [Conversation]:

的 return 值
let conversations = realm.objects(Conversation.self).sorted(by: {[=10=].lastUpdated < .lastUpdated})