Я создаю приложение iPhone с использованием Core Data.Как использовать NSFetchedResultsController и UISearchDisplayController
Прежде всего, имеет смысл использовать NSFetchedResultsController и UISearchDisplayController вместе для получения результата? Вы порекомендовали бы что-нибудь еще?
Я пытался довольно долго комбинировать NSFetchedResultController и UISearchDisplayController. Я думал об установке NSPredicate в методе (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchScope:(NSInteger)searchOption
в NSFetchedResultController UIViewController. Но это не так хорошо работает.
Итак, у вас есть идеи, как реализовать решение для моей проблемы? Благодарим вас за отправку ответов или ссылок на хорошие уроки.
EDIT
Вот мой код. Методы UISearchDisplayDelegate вызывают метод (void)filterContentForSearchText:(NSString*)searchText scope:(NSString*)scope
, который должен установить предикат в NSFetchedResultController. Я также добавил код NSFetchedResultController.
- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchString:(NSString *)searchString
{
[self filterContentForSearchText:searchString scope:
[[self.searchDisplayController.searchBar scopeButtonTitles] objectAtIndex:[self.searchDisplayController.searchBar selectedScopeButtonIndex]]];
// Return YES to cause the search result table view to be reloaded.
return YES;
}
- (BOOL)searchDisplayController:(UISearchDisplayController *)controller shouldReloadTableForSearchScope:(NSInteger)searchOption
{
[self filterContentForSearchText:[self.searchDisplayController.searchBar text] scope:
[[self.searchDisplayController.searchBar scopeButtonTitles] objectAtIndex:searchOption]];
// Return YES to cause the search result table view to be reloaded.
return YES;
}
- (void)filterContentForSearchText:(NSString*)searchText scope:(NSString*)scope
{
NSString *query = self.searchDisplayController.searchBar.text;
if (query && query.length) {
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"Name contains[cd] %@", query];
[fetchedResultsController.fetchRequest setPredicate:predicate];
}
NSError *error = nil;
if (![[self fetchedResultsController] performFetch:&error]) {
// Handle error
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
exit(-1); // Fail
}
}
- (NSFetchedResultsController *)fetchedResultsController {
if (fetchedResultsController != nil) {
return fetchedResultsController;
}
/*
Set up the fetched results controller.
*/
// Create the fetch request for the entity.
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
// Edit the entity name as appropriate.
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Event" inManagedObjectContext: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:@"timeStamp" ascending:NO];
NSArray *sortDescriptors = [[NSArray alloc] initWithObjects:sortDescriptor, nil];
[fetchRequest setSortDescriptors:sortDescriptors];
// Edit the section name key path and cache name if appropriate.
// nil for section name key path means "no sections".
NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:managedObjectContext sectionNameKeyPath:nil cacheName:@"Root"];
aFetchedResultsController.delegate = self;
self.fetchedResultsController = aFetchedResultsController;
[aFetchedResultsController release];
[fetchRequest release];
[sortDescriptor release];
[sortDescriptors release];
return fetchedResultsController;
}
Это решение работает, но является довольно дорогостоящим, заставляя контроллер получаемых результатов выполнять многочисленные поездки в постоянный магазин. Кроме того, если вы не собираетесь использовать кеш, зачем его указывать? Просто пройди ноль. Тем не менее, гораздо лучшее решение - это то, на что @psuedonick намекает: оставьте fetchedResultsController делать то, что он делает лучше всего, а именно управлял полученными результатами и использовал filterArrayUsingPredicate: on fetchedObjects, чтобы заполнить filterListContent. –
Чтобы уточнить, если вы используете решение, которое я описываю, вы должны * указать кеш: для этого нужно, чтобы получаемый им контроллер получал необходимые для хранения fetchedObjects, чтобы их можно было эффективно искать с помощью filterArrayUsingPredicate: –
Спасибо Elise, я думаю, ваш путь путь. Я завершу шаги по реализации контроллера отображения отображения в контроллере табличного представления в другом сообщении позже. Мне нужно столько времени, чтобы найти наилучшую практику по этому вопросу. – Philip007