1

在我的应用程序中,我经常在后台进行核心数据工作。

Testflight 一直在报告从 Core Data 获取当前登录环境对象的特定 fetch 的许多崩溃(我无法复制):

- (Environment *)getActiveEnvironment
{
    AppDelegate *ad = [AppDelegate sharedAppDelegate];
    NSManagedObjectContext *context = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSConfinementConcurrencyType];
    [context setParentContext:ad.managedObjectContext];
    
    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Environment" inManagedObjectContext:context];
    [fetchRequest setEntity:entity];
    
    NSError *error;

    //crash points to this line
    NSArray *items = [context executeFetchRequest:fetchRequest error:&error];
    
    Environment *tempE = nil;
    if(items.count > 0)
    {
        for(Environment *e in items)
        {
            if(!e.token && !e.url)
            {
                break;
            }
            
            Environment *mainContextTv = (Environment *)[ad.managedObjectContext objectWithID:e.objectID];
            if(e.activeValue)
                 tempE = mainContextTv;                
        }  
    }

    return tempE;
}

这是我从 Testflight 收到的崩溃堆栈:

0   Tower-iSales-Tab    0x002b37b2  testflight_backtrace
1   Tower-iSales-Tab    0x002b2e4a  TFSignalHandler
2   libsystem_platform.dylib    0x39144722  _sigtramp
3   CoreData    0x2e1b8cec  _perform
4   CoreData    0x2e1c30f4  -[NSManagedObjectContext(_NestedContextSupport) executeRequest:withContext:error:]
5   CoreData    0x2e13477a  -[NSManagedObjectContext executeFetchRequest:error:]
6   Tower-iSales-Tab    0x000c7082  -[MySingleton getActiveEnvironment] in MySingleton.m on Line 379
7   Tower-iSales-Tab    0x000c73f8  -[MySingleton getToken] in MySingleton.m on Line 416
8   Tower-iSales-Tab    0x0020c476  -[NetworkManager getPathForViewName:useTimeStamp:timeStamp:index:counter:] in NetworkManager.m on Line 699
9   Tower-iSales-Tab    0x0020aa5a  __70-[NetworkManager checkForDataFilesShouldEmptyQueue:isInitialDownload:]_block_invoke167 in NetworkManager.m on Line 415
10  libdispatch.dylib   0x3901b8fa  _dispatch_barrier_sync_f_invoke
11  Tower-iSales-Tab    0x0020a33e  __70-[NetworkManager checkForDataFilesShouldEmptyQueue:isInitialDownload:]_block_invoke113 in NetworkManager.m on Line 335
12  libdispatch.dylib   0x39017102  _dispatch_call_block_and_release
13  libdispatch.dylib   0x3901c7e4  _dispatch_root_queue_drain
14  libdispatch.dylib   0x3901c9d0  _dispatch_worker_thread2
15  libsystem_pthread.dylib 0x39146dfe  _pthread_wqthread
16  libsystem_pthread.dylib 0x39146cc3  start_wqthread

我读了这个答案,它建议将队列类型设置为NSPrivateQueueConcurrencyType,而不是NSConfinementConcurrencyType我可以做的,但这不是我可以轻松复制的崩溃,所以我想确定会导致这次崩溃。

该答案还建议使用该performBlockAndWait块进行后台提取,但在我的函数中我会怎么做?我需要返回一个Environment对象,我在整个应用程序中都使用了这个函数。

谢谢

4

2 回答 2

2

据我了解,您不必NSConfinementConcurrencyType在此处使用上下文,因为您-getActiveEnvironment仅从主线程调用。因此,您可以在主上下文 ( ad.managedObjectContext) 上执行所有操作。

然后应该只从主线程访问返回的对象。如果您想从其他线程访问它,您必须使用objectWithID:它在与该线程关联的上下文中获取它。

于 2013-10-24T14:56:49.557 回答
2

很可能,这个问题的原因是您正在NSManagedObjectContext从不同的线程访问它。

黄金法则:始终NSManagedObjectContext从您创建它的线程中访问一个。

所以你需要确保你的 MOC 从同一个线程被一致地访问。

我相信这些NSPrivateQueueConcurrencyType东西可能是偶然相关的,这样做可能会解决问题(取决于你的代码到底在做什么),但这可能不是这里的核心问题。

旁注:持久对象存储没有相同的要求并且是线程安全的(通常您有多个 MOC 使用相同的持久对象存储)。

于 2013-10-24T14:53:24.337 回答