在我的 iPhone 应用程序中,我使用具有两个实体(Item和Property)的简单核心数据模型:
项目
名称
属性
属性
名称
值
项
项目有一个属性(名称)和一个一对多的关系(属性)。它的反比关系是item。属性有两个属性对应的反比关系。
现在我想在两个级别的表格视图中显示我的数据。第一个列出所有项目;选择一行时,将推出一个新的UITableViewController在我的UinavigationController的堆栈上。新的 UITableView 应该显示所选项目的所有属性(即它们的名称)。
为了实现这一点,我使用了一个NSFetchedResultsController
存储在实例变量中。在第一层,像这样设置 NSFetchedResultsController 时一切正常:
-(NSFetchedResultsController *) fetchedResultsController {
if (fetchedResultsController) return fetchedResultsController;
// goal: tell the FRC to fetch all item objects.
NSFetchRequest *fetch = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Item" inManagedObjectContext:self.moContext];
[fetch setEntity:entity];
NSSortDescriptor *sort = [[NSSortDescriptor alloc] initWithKey:@"name" ascending:YES];
[fetch setSortDescriptors:[NSArray arrayWithObject:sort]];
[fetch setFetchBatchSize:10];
NSFetchedResultsController *frController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetch managedObjectContext:self.moContext sectionNameKeyPath:nil cacheName:@"cache"];
self.fetchedResultsController = frController;
fetchedResultsController.delegate = self;
[sort release];
[frController release];
[fetch release];
return fetchedResultsController;
}
但是,在二级UITableView上,我好像做错了什么。我以类似的方式实现了 fetchedresultsController:
-(NSFetchedResultsController *) fetchedResultsController {
if (fetchedResultsController) return fetchedResultsController;
// goal: tell the FRC to fetch all property objects that belong to the previously selected item
NSFetchRequest *fetch = [[NSFetchRequest alloc] init];
// fetch all Property entities.
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Property" inManagedObjectContext:self.moContext];
[fetch setEntity:entity];
// limit to those entities that belong to the particular item
NSPredicate *predicate = [NSPredicate predicateWithFormat:[NSString stringWithFormat:@"item.name like '%@'",self.item.name]];
[fetch setPredicate:predicate];
// sort it. Boring.
NSSortDescriptor *sort = [[NSSortDescriptor alloc] initWithKey:@"name" ascending:YES];
[fetch setSortDescriptors:[NSArray arrayWithObject:sort]];
NSError *error = nil;
NSLog(@"%d entities found.",[self.moContext countForFetchRequest:fetch error:&error]);
// logs "3 entities found."; I added those properties before. See below for my saving "problem".
if (error) NSLog("%@",error);
// no error, thus nothing logged.
[fetch setFetchBatchSize:20];
NSFetchedResultsController *frController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetch managedObjectContext:self.moContext sectionNameKeyPath:nil cacheName:@"cache"];
self.fetchedResultsController = frController;
fetchedResultsController.delegate = self;
[sort release];
[frController release];
[fetch release];
return fetchedResultsController;
}
现在越来越奇怪了。上面的NSLog
语句返回了所选项目的正确数量的属性。但是, UITableViewDelegate 方法告诉我没有属性:
-(NSInteger) tableView:(UITableView *)table numberOfRowsInSection:(NSInteger)section {
id <NSFetchedResultsSectionInfo> sectionInfo = [[self.fetchedResultsController sections] objectAtIndex:section];
NSLog(@"Found %d properties for item \"%@\". Should have found %d.",[sectionInfo numberOfObjects], self.item.name, [self.item.properties count]);
// logs "Found 0 properties for item "item". Should have found 3."
return [sectionInfo numberOfObjects];
}
相同的实现在第一级上运行良好。
越来越奇怪了。我实现了某种 UI 来添加属性。我通过创建一个新的Property实例Property *p = [NSEntityDescription insertNewObjectForEntityForName:@"Property" inManagedObjectContext:self.moContext];
,设置关系并调用[self.moContext save:&error]
。这似乎error
仍然有效nil
,并且对象被保存(我可以在记录Item实例时看到属性的数量,见上文)。但是,不会触发委托方法。这在我看来是因为 fetchRequest(Controller) 可能搞砸了。
有任何想法吗?我搞砸了第二个获取请求吗?这是为特定实例获取多对多关系中所有实体的正确方法吗?