Как использовать ManagedObjectContext с потоками - PullRequest
1 голос
/ 24 декабря 2010

Это, вероятно, очень прямолинейное приложение, но я новичок в Objective-C (из Java), и все ошибки управления памятью и "EXC_BAD_ACCESS" разбивают мне сердце.

У меня есть нормальное приложение NavigationController для iPhone с Core Data. в AppDelegate NSManagedObjectContext создается и передается в RootViewController. Представление просматривается прямо из основного потока, чтобы заполнить таблицу, и это, кажется, работает нормально.

Приложение является своего рода читателем RSS-типа, поэтому, как только приложение запускается, я запускаю поток для извлечения новых данных и обновления представления:

-(void)updateData:(id)sender {
 UIActivityIndicatorView *activityIndicator =
    [[UIActivityIndicatorView alloc] initWithFrame:CGRectMake(0, 0, 20, 20)];
 [activityIndicator startAnimating];
 UIBarButtonItem *activityItem =
    [[UIBarButtonItem alloc] initWithCustomView:activityIndicator];
 [activityIndicator release];
 self.navigationItem.leftBarButtonItem = activityItem;
 [activityItem release];

 // Start thread to update the data
 [NSThread detachNewThreadSelector:@selector(doUpdateData) toTarget:self withObject:nil];
}

-(void)doUpdateData{
 NSLog(@"Update data Thread (in 5 sec.)");
 [NSThread sleepForTimeInterval:5];

 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];

 DataManager *data = [[DataManager alloc] initWithContext:managedObjectContext];
 [data updateData];
 [data release];
 data=nil;

 [self performSelectorOnMainThread:@selector(finishUpdateData) withObject:nil waitUntilDone:NO]; 
 [pool release];
}

-(void)finishUpdateData{
 self.navigationItem.leftBarButtonItem = updateBttn;
 DataManager *data = [[DataManager alloc] initWithContext:managedObjectContext];
 objects = [data getArticles];
 [data release];
 data=nil;
 NSLog(@"Amount of records after update: %d", [objects count]);
 [self.tableView reloadData]; 
}

Проблема в том, что это не работает. В DataManager необходимо получить первые настройки, и как только NSEntityDescription будет создано, я получу «EXC_BAD_ACCESS»:

- (NSFetchedResultsController *)fetchedResultsController {
    // Set up the fetched results controller if needed.
    if (fetchedResultsController == nil) {
        NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
        NSEntityDescription *entity = [NSEntityDescription entityForName:@"Setting" inManagedObjectContext:managedObjectContext];
        [fetchRequest setEntity:entity];

        NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"key" ascending:YES];
        NSArray *sortDescriptors = [[NSArray alloc] initWithObjects:sortDescriptor, nil];

        [fetchRequest setSortDescriptors:sortDescriptors];
  [fetchRequest setFetchLimit:1]; 
        .
        NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:managedObjectContext sectionNameKeyPath:nil cacheName:nil];

        aFetchedResultsController.delegate = self;
        self.fetchedResultsController = aFetchedResultsController;

        [aFetchedResultsController release];
        [fetchRequest release];
        [sortDescriptor release];
        [sortDescriptors release];
    }

 return fetchedResultsController;
} 

Я полагаю, что указатель на ManagedObjectContext неверен из-за запуска в другом потоке и в пуле памяти. Итак, как вы создаете такое приложение, если это проблема), как мне получить ссылку на оригинальный формат ManagedObjectContext, который он представляет?

[EDIT] Я также пытался использовать

iDomsAppDelegate *appDelegate = (iDomsAppDelegate *)[[UIApplication sharedApplication] delegate];
DataManager *data = [[DataManager alloc] initWithContext:appDelegate.managedObjectContext];

в doUpdateData (как подсказано другими сообщениями), но это дает тот же результат

1 Ответ

8 голосов
/ 24 декабря 2010

Контексты управляемых объектов не являются потокобезопасными. Рекомендации Apple указывают на то, что должен иметь отдельный экземпляр NSManagedObjectContext для потока.

...