自定义核心数据SectionNameKeyPath

时间:2013-04-08 16:06:08

标签: ios core-data transient sections

我是核心数据新手,我正试图找出如何在sectionNameKeyPath中创建自定义NSFetchedResultsController。我有一个名为acctPeriod的托管对象。这是NSString。我想根据此字段的前4个字符创建部分。前4个字符代表会计期间的年份,不需要保存。

我已浏览过这个网站并看过有关瞬态属性的帖子,但我似乎无法让它们工作。基本上我想要这个,然后为periodYear分配sectionNameKeyPath

@dynamic periodYear;

-(NSString *)periodYear
{
    return [self.acctPeriod substringToIndex:4];
}

任何帮助都将不胜感激。

**更新: 使用Martin R.回答,我能够按预期工作。

- (NSFetchedResultsController *)fetchedResultsController
{
if (_fetchedResultsController != nil) {
    return _fetchedResultsController;
}

NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];

// Edit the entity name as appropriate.
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Billing" inManagedObjectContext:self.managedObjectContext];
[fetchRequest setEntity:entity];

// Set the batch size to a suitable number.
[fetchRequest setFetchBatchSize:20];

// Edit the sort key as appropriate.
NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"acctPeriod" ascending:NO];
NSArray *sortDescriptors = @[sortDescriptor];

//Predicate
NSPredicate *pred = [NSPredicate predicateWithFormat:@"clients = %@", self.client];
NSLog(@"%@",pred);

//[fetchRequest setResultType:NSDictionaryResultType];
//[fetchRequest setReturnsDistinctResults:YES];

[fetchRequest setPredicate:pred];
[fetchRequest setSortDescriptors:sortDescriptors];

NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:self.managedObjectContext sectionNameKeyPath:@"periodYear" cacheName:nil];
aFetchedResultsController.delegate = self;
self.fetchedResultsController = aFetchedResultsController;

NSError *error = nil;
if (![self.fetchedResultsController performFetch:&error])
{
    NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
    abort();
}

return _fetchedResultsController;  
}

2 个答案:

答案 0 :(得分:28)

以下内容应该有效:实施periodYear方法(将使用该方法) 在托管对象子类的类扩展中作为“节名称键路径”):

@interface Event (AdditionalMethods)
- (NSString *)periodYear;
@end

@implementation Event (AdditionalMethods)
- (NSString *)periodYear {
    return [self.acctPeriod substringToIndex:4];
}
@end

确保将acctPeriod用作作为获取请求的第一个(或唯一的)排序描述符

NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"acctPeriod" ascending:YES];
NSArray *sortDescriptors = @[sortDescriptor];
[fetchRequest setSortDescriptors:sortDescriptors];

使用periodYear作为sectionNameKeyPath获取的结果控制器:

NSFetchedResultsController *_fetchedResultsController = [[NSFetchedResultsController alloc]
                  initWithFetchRequest:fetchRequest 
                   managedObjectContext:self.managedObjectContext 
                     sectionNameKeyPath:@"periodYear"
                              cacheName:nil];
_fetchedResultsController.delegate = self;
self.fetchedResultsController = _fetchedResultsController;

最后添加默认的titleForHeaderInSection方法:

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

或者,您可以将periodYear定义为托管对象的瞬态属性。 在这种情况下,它也不会存储在数据库中,但可以按需要计算和缓存值的方式实现。

Apple Developer Library中的DateSectionTitles示例项目演示了这是如何工作的。

答案 1 :(得分:8)

我建议不要使用瞬态属性作为sectionNameKeyPath,因为它会导致获取请求获取的所有对象出错,因此该属性可以用作节路径。
您最好定义year的持久属性,并将其用作sectionNameKeyPath 将FRC sectionNameKeyPath设置为year,如下所示:

[[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest
                                    managedObjectContext:self.managedObjectContext
                                      sectionNameKeyPath:@"year"
                                               cacheName:nil/*your chioce*/];

将表格名称显示为表格中的标题,执行:

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