在计算 属性 上对 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
装饰。我尝试添加它,擦除我的应用程序安装,重新编译并再次测试,但我仍然遇到相同的异常。
也许无法保留计算的 属性(因为,好吧,它是计算的!)但是是否有另一种方法可以指定我的对话结果集合应按此 属性 排序?
正如您所怀疑的,您无法持久化计算的 属性 并且 Realm
的 sorted(byKeyPath:,ascending:)
方法仅适用于持久化属性。在保持自动更新 Results
.
的同时,无法根据计算的 属性 对 Results
实例进行排序
但是,如果您不需要 Results
的自动更新特性,则有一个解决方法。您可以使用 Swift 的 sorted
方法,该方法将闭包作为其排序参数,这将导致类型为 [Conversation]
:
的 return 值
let conversations = realm.objects(Conversation.self).sorted(by: {[=10=].lastUpdated < .lastUpdated})
我在我的 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
装饰。我尝试添加它,擦除我的应用程序安装,重新编译并再次测试,但我仍然遇到相同的异常。
也许无法保留计算的 属性(因为,好吧,它是计算的!)但是是否有另一种方法可以指定我的对话结果集合应按此 属性 排序?
正如您所怀疑的,您无法持久化计算的 属性 并且 Realm
的 sorted(byKeyPath:,ascending:)
方法仅适用于持久化属性。在保持自动更新 Results
.
Results
实例进行排序
但是,如果您不需要 Results
的自动更新特性,则有一个解决方法。您可以使用 Swift 的 sorted
方法,该方法将闭包作为其排序参数,这将导致类型为 [Conversation]
:
let conversations = realm.objects(Conversation.self).sorted(by: {[=10=].lastUpdated < .lastUpdated})