自定义核心数据SectionNameKeyPath
我是新的核心数据,并试图找出如何在我的sectionNameKeyPath
中创build一个自定义sectionNameKeyPath
。 我有一个名为acctPeriod
属性的托pipe对象。 这是一个NSString
。 我想创build基于该字段的前4个字符的部分。 前4个字符代表会计期间的年份,不需要保存。
我经历了这个网站,看到关于瞬态属性的post,但我似乎无法让他们工作。 基本上我想这个,然后为我的sectionNameKeyPath
分配sectionNameKeyPath
。
@dynamic periodYear; -(NSString *)periodYear { return [self.acctPeriod substringToIndex:4]; }
任何帮助,将不胜感激。
**更新:使用马丁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; }
以下内容应该可以工作:在托pipe对象子类的类扩展中实现periodYear
方法(将用作“段名称关键path”):
@interface Event (AdditionalMethods) - (NSString *)periodYear; @end @implementation Event (AdditionalMethods) - (NSString *)periodYear { return [self.acctPeriod substringToIndex:4]; } @end
确保acctPeriod
被用作获取请求的第一个(或唯一)sorting描述符:
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
定义为pipe理对象的瞬态属性 。 在这种情况下,它也不会被存储在数据库中,而是可以按照需要计算和caching的值来实现。
Apple开发者库的DateSectionTitles示例项目演示了这是如何工作的。
我build议不要使用transient属性作为sectionNameKeyPath
因为这会导致获取请求获取的所有对象都正常,因此属性可以用作节path。
你最好定义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]; }