自定义部分名称崩溃NSFetchedResultsController

我有一个带有dueDate属性的托管对象。 而不是显示使用一些丑陋的日期字符串作为我的UITableView的部分标题,我创建了一个名为“类别”的瞬态属性,并像这样定义它:

- (NSString*)category
{
    [self willAccessValueForKey:@"category"];

    NSString* categoryName;
    if ([self isOverdue])
    {
        categoryName = @"Overdue";
    }
    else if ([self.finishedDate != nil])
    {
        categoryName = @"Done";
    }
    else
    {
        categoryName = @"In Progress";
    }

    [self didAccessValueForKey:@"category"];
    return categoryName;
}

这里是NSFetchedResultsController的设置:

NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Task"
                                          inManagedObjectContext:managedObjectContext];
[fetchRequest setEntity:entity];

NSMutableArray* descriptors = [[NSMutableArray alloc] init];
NSSortDescriptor *dueDateDescriptor = [[NSSortDescriptor alloc] initWithKey:@"dueDate"
                                                                  ascending:YES];
[descriptors addObject:dueDateDescriptor];
[dueDateDescriptor release];
[fetchRequest setSortDescriptors:descriptors];

fetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:managedObjectContext sectionNameKeyPath:@"category" cacheName:@"Root"];

该表最初显示正常,显示其dueDate未在标题为“进行中”的部分中传递的未完成项目。 现在,用户可以点击表格视图中的一行,将新的详细信息视图推送到导航堆栈上。 在这个新视图中,用户可以点击一个按钮来指示该项目现在是“完成”。 这里是按钮的处理程序(self.task是托管对象):

- (void)taskDoneButtonTapped
{
    self.task.finishedDate = [NSDate date];
}

只要“finishedDate”属性的值发生变化,我就会遇到这个异常:

2010-03-18 23:29:52.476 MyApp[1637:207] Serious application error.  Exception was caught during Core Data change processing: no section named 'Done' found with userInfo (null)
2010-03-18 23:29:52.477 MyApp[1637:207] *** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'no section named 'Done' found'

我设法弄清楚,当前被新细节视图隐藏的UITableView试图更新它的行和部分,因为NSFetchedResultsController被告知数据集中的某些内容发生了变化。 这里是我的表更新代码(从Core Data Recipes示例或CoreBooks示例中复制 - 我不记得哪个):

- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller
{
    [self.tableView beginUpdates];
}

- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject atIndexPath:(NSIndexPath *)indexPath forChangeType:(NSFetchedResultsChangeType)type newIndexPath:(NSIndexPath *)newIndexPath
{
    switch(type)
    {
        case NSFetchedResultsChangeInsert:
            [self.tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeDelete:
            [self.tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeUpdate:
            [self configureCell:[self.tableView cellForRowAtIndexPath:indexPath] atIndexPath:indexPath];
            break;

        case NSFetchedResultsChangeMove:
            [self.tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
            // Reloading the section inserts a new row and ensures that titles are updated appropriately.
            [self.tableView reloadSections:[NSIndexSet indexSetWithIndex:newIndexPath.section] withRowAnimation:UITableViewRowAnimationFade];
            break;
    }
}

- (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id <NSFetchedResultsSectionInfo>)sectionInfo atIndex:(NSUInteger)sectionIndex forChangeType:(NSFetchedResultsChangeType)type
{
    switch(type)
    {
        case NSFetchedResultsChangeInsert:
            [self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
            break;

        case NSFetchedResultsChangeDelete:
            [self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
            break;
    }
}

- (void)controllerDidChangeContent:(NSFetchedResultsController *)controller
{
    [self.tableView endUpdates];
}

我在这些函数中加入了断点,并发现只调用了controllerWillChange。 在两个控制器之一之前抛出异常:didChangeObject:atIndexPath:forChangeType:newIndex或controller:didChangeSection:atIndex:forChangeType被调用。

此时我卡住了。 如果我将我的sectionNameKeyPath更改为“dueDate”,那么一切正常。 我认为这是因为dueDate属性永远不会改变,而在finishedDate属性更改后回读时类别会有所不同。

请帮忙!

更新:

这是我的UITableViewDataSource代码:

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
    return [[self.fetchedResultsController sections] count];
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    id <NSFetchedResultsSectionInfo> sectionInfo = [[self.fetchedResultsController sections] objectAtIndex:section];
    return [sectionInfo numberOfObjects];
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"Cell";

    UITableViewCell *cell = [self.tableView dequeueReusableCellWithIdentifier:CellIdentifier];
    if (cell == nil)
    {
        cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier] autorelease];
    }

    [self configureCell:cell atIndexPath:indexPath];    

    return cell;
}

- (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section
{
    id <NSFetchedResultsSectionInfo> sectionInfo = [[self.fetchedResultsController sections] objectAtIndex:section];    
    return [sectionInfo name];
}

崩溃是由于NSFetchedResultsController不知道“完成”类别而导致崩溃。 我在其他问题中几次看到过这次崩溃,并且我建议每次都向苹果公司提交雷达票。 这是NSFetchedResultsController一个错误。


它看起来像你的问题在于你用来提供sectionNameKeyPath的“类别”瞬态属性。 sectionNameKeyPath必须与主排序描述符相同。 就你而言,这意味着所有“逾期”任务必须具有早于所有“完成”任务的日期必须早于所有“正在进行”任务的日期。 可以构建一个场景,其中“完成”任务的“到期日期”任务之后有dueDate,或者出现在“逾期”任务之前。 这个场景打破了sectionNameKeyPath的顺序要求,并导致NSFetchedResultsController抛出一个NSInternalConsistencyException异常。

我为您的问题提出了一个解决方案,不涉及滚动您自己的数组,然后必须将其分割成多个部分。 在模型中创建一个整数属性,将0映射到“过期”,1映射到“完成”,将2映射到“正在进行”。 将它作为NSFetchRequest中的主要排序描述符并按升序对此属性进行排序。 向NSFetchRequest添加一个辅助排序描述符,以升序排序dueDate属性。 修改您的类别方法,从您在上面创建的整数属性派生类别名称,并将其用作您的sectionNameKeyPath。 您将需要更新整数属性,以便在任务从正在进行到逾期完成时更新任务等。

链接地址: http://www.djcxy.com/p/36097.html

上一篇: Custom Section Name Crashing NSFetchedResultsController

下一篇: How do you handle section inserts with an NSFetchedResultsController?