Базовые данные NSObjectInaccessibleException NSManagedObject признан недействительным - PullRequest
1 голос
/ 28 августа 2011

У меня есть некоторый код, который загружает уроки как JSON, анализирует их и помещает их в основные данные. Затем они отображаются в UITableView. В настоящее время, когда у пользователя много уроков, время от времени соединение прерывается. Поэтому я пытаюсь анализировать уроки по мере их поступления (используя SBJson) и добавлять их в просмотр таблицы по одному.

Код для двух в основном одинаков, но новый код приводит к сбою, когда включается содержимое tableView, с ошибкой

"Terminating app due to uncaught exception 
'NSObjectInaccessibleException', reason: 'The NSManagedObject with ID:0x5ad0570 <x-coredata://A21AC71F-175B-423D-BF7D-C67BEE094460/Lessons/p18> has been invalidated.'"

Я хотел бы знать, в чем разница между этими двумя списками кодов, которые могут быть причиной этой ошибки. Исходный код создает каждый объект данных ядра в цикле, но новый код создает каждый объект данных ядра по мере его загрузки. listViewArray - это массив, который используется для заполнения UITableView.

Я использую SBJsonStreamParser и SBJsonStreamParserAdapter для анализа Json, когда он поступает.

У меня есть рабочая реализация (не показана), которая в основном вызывает исходный код ниже каждый раз при получении нового объекта (каждый раз проходя через полный цикл принятых объектов). Я хочу знать, что является причиной ошибки, а не просто заставить что-то работать.

Это оригинальный не потоковый код, вызываемый в connectionDidFinishLoading:

    NSMutableArray *tempListArray = [[NSMutableArray alloc] initWithArray:jsonStreamedData];

    if (listViewArray)
        [listViewArray release];
    listViewArray = [[NSMutableArray alloc] init];

    if(![tempListArray count]){
        UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"Message" message:@"No active lessons " delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
        [alertView show];
        [alertView release];
    }
    else {
        MyAppDelegate *appDelegate = (MyAppDelegate *)[[UIApplication sharedApplication] delegate];
        NSError *error = nil;
        [appDelegate.managedObjectContext reset];

        NSManagedObjectContext *managedObjectContext = appDelegate.managedObjectContext;
        for (int i = 0; i < [tempListArray count]; i++) {

            NSFetchRequest *checkRequest = [[NSFetchRequest alloc] init];
            NSEntityDescription *lessonEntity = [NSEntityDescription entityForName:@"Lessons" inManagedObjectContext:managedObjectContext];
            [checkRequest setEntity:lessonEntity];
            NSPredicate *langPredicate = [NSPredicate predicateWithFormat:@"(language = %@)", appDelegate.currentLanguage];
            NSPredicate *userPredicate = [NSPredicate predicateWithFormat:@"(username = %@)", appDelegate.userName];
            NSPredicate *idPredicate = [NSPredicate predicateWithFormat:@"(content_Id = %@)", [[tempListArray objectAtIndex:i] valueForKey:@"id"]];
            [checkRequest setPredicate:[NSCompoundPredicate andPredicateWithSubpredicates:[NSArray arrayWithObjects:langPredicate, userPredicate, idPredicate, nil]]];

            NSArray *checkResults = [managedObjectContext executeFetchRequest:checkRequest error:&error];
            [checkRequest release];

            NSMutableDictionary *tempDict = [[NSMutableDictionary alloc] init];
            if ([checkResults count]) {
                Lessons *lessonObj = [checkResults objectAtIndex:0];
                lessonObj.cards_count = [[tempListArray objectAtIndex:i] valueForKey:@"cards_count"];
                lessonObj.mTitle = [[tempListArray objectAtIndex:i] valueForKey:@"title"];
                lessonObj.sound_Url = [[tempListArray objectAtIndex:i] valueForKey:@"audio_url"];
                lessonObj.mId = [NSNumber numberWithInt:i];
                [tempDict setValue:lessonObj forKey:@"lesson"];
                [tempDict setValue: [[tempListArray objectAtIndex:i]  objectForKey:@"image_url"] forKey:@"image_url"];
                [listViewArray addObject:tempDict];

            }
            else {

                Lessons *newLesson = (Lessons *)[NSEntityDescription insertNewObjectForEntityForName:@"Lessons" inManagedObjectContext:appDelegate.managedObjectContext];
                newLesson.cards_count = [[tempListArray objectAtIndex:i] valueForKey:@"cards_count"];
                newLesson.mTitle = [[tempListArray objectAtIndex:i] valueForKey:@"title"];
                newLesson.sound_Url = [[tempListArray objectAtIndex:i] valueForKey:@"audio_url"];
                newLesson.content_Id = [[tempListArray objectAtIndex:i] valueForKey:@"id"];
                newLesson.username = appDelegate.userName;
                newLesson.language = appDelegate.currentLanguage;
                newLesson.mId = [NSNumber numberWithInt:i];
                [tempDict setValue:newLesson forKey:@"lesson"];
                [tempDict setValue: [[tempListArray objectAtIndex:i]  objectForKey:@"image_url"] forKey:@"image_url"]; 
                [listViewArray addObject:tempDict];

            }
            [tempDict release];
            tempDict = nil;
        }


        if (![appDelegate.managedObjectContext save:&error]) {
            NSLog(@"Core Data Error - %@", [error localizedDescription]);

        }   

        //      NSMutableArray *tempArray = [NSMutableArray arrayWithArray:listViewArray];
        //      [listViewArray removeAllObjects];
        //      [listViewArray addObjectsFromArray:[[tempArray reverseObjectEnumerator] allObjects]];
        //      tempArray = nil;
    }
    [tempListArray release];
}
[mListsTableView reloadData];

А вот код сбоя, вызываемый в анализаторе: foundObject: Код цикла был удален, поскольку он вызывается каждый раз, когда загружается новый объект Json.

    [jsonStreamedData addObject:dict];
    if (!listViewArray)
        listViewArray = [[NSMutableArray alloc] init];

    MyAppDelegate *appDelegate = (MyAppDelegate *)[[UIApplication sharedApplication] delegate];
    NSError *error = nil;
    [appDelegate.managedObjectContext reset];

    NSManagedObjectContext *managedObjectContext = appDelegate.managedObjectContext;

    NSFetchRequest *checkRequest = [[NSFetchRequest alloc] init];
    NSEntityDescription *lessonEntity = [NSEntityDescription entityForName:@"Lessons" inManagedObjectContext:managedObjectContext];
    [checkRequest setEntity:lessonEntity];
    NSPredicate *langPredicate = [NSPredicate predicateWithFormat:@"(language = %@)", appDelegate.currentLanguage];
    NSPredicate *userPredicate = [NSPredicate predicateWithFormat:@"(username = %@)", appDelegate.userName];
    NSPredicate *idPredicate = [NSPredicate predicateWithFormat:@"(content_Id = %@)", [dict valueForKey:@"id"]];
    [checkRequest setPredicate:[NSCompoundPredicate andPredicateWithSubpredicates:[NSArray arrayWithObjects:langPredicate, userPredicate, idPredicate, nil]]];

    NSArray *checkResults = [managedObjectContext executeFetchRequest:checkRequest error:&error];
    [checkRequest release];

    NSMutableDictionary *tempDict = [[NSMutableDictionary alloc] init];

    if ([checkResults count]) {
        Lessons *lessonObj = [checkResults objectAtIndex:0];
        lessonObj.cards_count = [dict valueForKey:@"cards_count"];
        lessonObj.mTitle = [dict  valueForKey:@"title"];
        lessonObj.sound_Url = [dict valueForKey:@"audio_url"];
        lessonObj.mId = [NSNumber numberWithInt:jsonStreamedData.count - 1]; // This should be equivalent to i from the loop in the first code
        [tempDict setValue:lessonObj forKey:@"lesson"];
        [tempDict setValue: [dict objectForKey:@"image_url"] forKey:@"image_url"];
        [listViewArray addObject:tempDict];

    }
    else {

        Lessons *newLesson = (Lessons *)[NSEntityDescription insertNewObjectForEntityForName:@"Lessons" inManagedObjectContext:appDelegate.managedObjectContext];
        newLesson.cards_count = [dict valueForKey:@"cards_count"];
        newLesson.mTitle = [dict valueForKey:@"title"];
        newLesson.sound_Url = [dict valueForKey:@"audio_url"];
        newLesson.content_Id = [dict valueForKey:@"id"];
        newLesson.username = appDelegate.userName;
        newLesson.language = appDelegate.currentLanguage;
        newLesson.mId = [NSNumber numberWithInt:jsonStreamedData.count - 1];
        [tempDict setValue:newLesson forKey:@"lesson"];
        [tempDict setValue: [dict  objectForKey:@"image_url"] forKey:@"image_url"]; 
        [listViewArray addObject:tempDict];

    }

    [tempDict release];
    tempDict = nil;

    if (![appDelegate.managedObjectContext save:&error]) {
        ALog(@"Core Data Error - %@", [error localizedDescription]);
    }   

    //  NSMutableArray *tempArray = [NSMutableArray arrayWithArray:listViewArray];
    //  [listViewArray removeAllObjects];
    //  [listViewArray addObjectsFromArray:[[tempArray reverseObjectEnumerator] allObjects]];
    //  tempArray = nil;
//[self getListsLocally];
[mListsTableView reloadData];

Наконец, вот фактическая часть, которая вылетает при использовании второго листинга, в tableView: cellForRowAtIndexPath: Кстати, она вылетает, когда строка == 1, а не строка == 0. По какой-то причине строка 0 в порядке .. Конечно, у него никогда не будет возможности загрузить другие строки.

        titleLabel.text = [[[listViewArray objectAtIndex:indexPath.row] valueForKey:@"lesson"] valueForKey:@"mTitle"]; // CRASH!
        labelCards.text = [NSString stringWithFormat:@"%@ Cards", [[[listViewArray objectAtIndex:indexPath.row]  valueForKey:@"lesson"] valueForKey:@"cards_count"]];

        if([[listViewArray objectAtIndex:indexPath.row] objectForKey:@"userImageObj"] == nil){
            mImageView.backgroundColor = [UIColor grayColor];
            if ([[listViewArray objectAtIndex:indexPath.row] objectForKey:@"isThreadLaunched"] == nil) {
                [NSThread detachNewThreadSelector:@selector(loadImagesInBackground:) toTarget:self withObject:[NSNumber numberWithInt:indexPath.row]];
                [[listViewArray objectAtIndex:indexPath.row] setObject:@"Yes" forKey:@"isThreadLaunched"];
            }
        }else {
            mImageView.image = [[listViewArray objectAtIndex:indexPath.row] objectForKey:@"userImageObj"];
        }

Ответы [ 2 ]

3 голосов
/ 30 августа 2011

Недействительность объекта, скорее всего, происходит, когда вы вызываете reset для managedObjectContext непосредственно перед выполнением выборки. Вызов reset делает недействительными объекты в памяти, но не удаляет их до сохранения. Если недействительный управляемый объект сохраняется другим объектом, например массивом, он сохраняется после сохранения в недействительной форме. Когда вы запускаете выборку, выборка возвращает недействительные объекты, которые вызывают ошибку при попытке доступа к одному из их атрибутов

reset предназначен для вызова при использовании с менеджером отмены. Это не общий вызов "стереть контекст". Если вы хотите удалить существующие объекты, вам нужно их извлечь и явно удалить.

У вашего кода также есть другие проблемы. Вы вызываете release для массива checkRequest, даже если вы его не создавали. Это может привести к случайному исчезновению массива. Точно так же listViewArray представляется свойством класса, но вы никогда не используете формы доступа, например self.listViewArray для обеспечения надлежащего удержания.

1 голос
/ 29 августа 2011

Скорее всего, причина в том, что анализатор не работает в главном потоке, и вы используете один и тот же ManagedObjectContext в основном потоке и в синтаксическом анализаторе - вы не можете этого сделать - гарантированно вызывает все виды странного поведения.

Вам необходимо создать новый ManagedObjectContext в парсере и связать его с persistentStore ManagedObjectContext основного потока

Платформа CoreData совершенно ясна в этом - вы не можете совместно использовать ManagedObjectContext через границы потоков

...