13

在此处输入图像描述- 我有一个Item实体和一个Tag实体。- 项目可以有多个标签,标签可以链接到多个项目(多对多关系)。- 关系是“有序关系”(在IOS5中使用有序关系)两种方式。

我想获取给定项目的所有子标签

我使用以下获取请求:

NSFetchRequest* request = [NSFetchRequest fetchRequestWithEntityName:@"Item"];

// Fetch all items that have a given tag
Tag* myTag = ....;
request.predicate = [NSPredicate predicateWithFormat:@"ANY tag == %@", myTag];

// This is my attempt to get the correct sort ordering (it crashes)
NSSortDescriptor *sortDescriptor = [NSSortDescriptor sortDescriptorWithKey:@"tag"
                                                                 ascending:YES];
request.sortDescriptors = @[sortDescriptor];

上面的排序描述符返回数据(我假设它是按某种顺序)但随后崩溃:

[_NSFaultingMutableOrderedSet compare:]: unrecognized selector sent to instance 0x10b058f0
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: 
'-[_NSFaultingMutableOrderedSet compare:]: unrecognised selector sent to instance 0x10b058f0'

如果我给出一个空的排序描述符数组,那么我不会崩溃,但结果不是有序的。

如何使用 ios5 中的有序关系功能在多对多关系中正确实现排序描述符?

4

1 回答 1

16

我认为您可能误解了 Core Data 中有序关系的含义。根据这个问题,“有序”并不意味着“关于某些属性” - 相反,“有序”意味着“保留用户的订单”(或其他看似任意的特定订单)。一些例子:

  • 有序关系对于配方中的步骤(来自 Apple 的核心数据发布说明的示例)是有益的 - 这些步骤不一定针对它们的任何属性(例如它们的文本)进行排序,并且必须维护一个 "如果 Core Data 没有进行排序,则它们自己的 index" 或 "order" 属性。
  • 有序关系不适用于名称列表 - 例如,用户可以选择按名字或姓氏排序,而名称集合在内部没有真正的内在排序。唯一重要的是显示顺序 - 数据本身不需要维护该顺序。(Apple 文档中的经验法则是,如果用户可以在排序说明符之间进行选择,则不应在 Core Data 中对关系进行排序。)

在您的情况下,您尝试将排序描述符应用于您正在获取的一组有序集tags属性的值)。这是您崩溃的原因- NSOrderedSet 子类(_NSFaultingMutableOrderedSet在您的情况下)没有实现-compare:排序描述符用于对它们进行排序的方法,因此您会遇到该异常。

我认为对您来说一个好的解决方案是根本不使用排序描述符。相反,如果您想更改保留的顺序以显示给用户,请获取tags某个项目的 NSOrderedSet 值,然后使用您自己的排序从中获取一个数组-sortedArrayUsingComparator:。这将使数据在 Core Data 中保持正确的顺序,但让您以编程方式自行重新排序。

在更高的层次上,你可能还想考虑你是否真的需要在你的关系中保持秩序。它会导致显着的性能损失,您真正得到的只是将标签添加到项目的原始顺序(反之亦然)。如果您打算在每次显示标签时更改该顺序,那么您会在没有真正的好处的情况下对性能造成影响 - 使您的关系无序,然后以编程方式继续排序或使用聚合键上的排序描述符,例如tags.@count,正如评论中所建议的那样。

于 2012-09-11T16:27:44.687 回答