Скажем, у меня есть объекты, организованные в иерархию, где Parent является корневым объектом, а Child является подклассом Parent. Я хотел бы настроить NSArrayController для извлечения только объектов Parent, но не Child.
Если вы установите Имя объекта контроллера массива в Интерфейсном Разработчике на Parent, он выбирает все объекты Parent и Child. Первоначально я пытался установить предикат выборки контроллера массива в Interface Builder на:
entity.name == "Parent"
Это сработало для хранилища XML, но когда я переключился на хранилище SQLite, он больше не работал. Я получаю следующую ошибку:
keypath entity.name not found in entity <NSSQLEntity xxx>
В качестве обходного пути я настраиваю предикат фильтра (с тем же предикатом entity.name, что и выше) в моем awakeFromNib, чтобы фильтровать только родительские объекты. По-видимому, этот предикат действителен, когда объекты находятся в памяти, но вы не можете использовать его в предикате выборки с поддержкой SQL.
Есть ли способ получить только объекты Parent, но не объекты Child, используя предикат выборки, который работает с хранилищем SQLite? Кажется расточительным привлекать объекты, которые вы собираетесь игнорировать только с помощью предиката фильтра.





Leopard представил свойство includesSubentities для NSFetchRequest именно для этой цели. Вам нужно будет создать подкласс своего NSObjectController или NSArrayController, чтобы предоставить запрос на выборку, который он будет использовать через свойство defaultFetchRequest, или изменить используемый запрос выборки, переопределив его метод -fetchWithRequest:....
Я пробовал использовать includesSubentities, но он не работал полностью. Оказывается, изменения в подчиненных сущностях приводят к обновлению содержимого контроллера массива без выполнения выборки, если для параметра «автоматически подготавливает контент» установлено значение «Да», таким образом обходя настраиваемый предикат выборки. Трассировка показывает, что setContent: вызывается в ответ на уведомления MOC.
Я нашел единственный способ сделать это надежно - использовать предикат фильтра.
В дополнение к ответу Дэйва Дрибина выше ... также убедитесь, что вы установили для clearsFilterPredicateOnInsertion значение NO, чтобы ваш фильтр не очищался, как только вы что-то вставляете или удаляете (вы можете сделать это в коде или IB).