应用程序突然崩溃每次从"You have illegally mutated the NSFetchedResultsController's fetch request"启动,但解决方案令人不安



每次运行应用程序时,我都会突然收到以下错误:

核心数据:致命错误:部分信息的持久缓存与当前配置不匹配。 您非法更改了 NSFetchedResultsController 的获取请求、其谓词或其排序描述符,而未禁用缓存或使用 +deleteCacheWithName:

下面是调用堆栈中的前十项:

*** First throw call stack:
(
    0   CoreFoundation                      0x04b835e4 __exceptionPreprocess + 180
    1   libobjc.A.dylib                     0x022c68b6 objc_exception_throw + 44
    2   CoreData                            0x020a1b41 -[NSFetchedResultsController performFetch:] + 913
    3   Syllable                            0x00036aa9 -[RootViewController fetchedResultsController] + 777
    4   Syllable                            0x0003772b -[RootViewController tableView:numberOfRowsInSection:] + 91
    5   UIKit                               0x00d1d240 -[UISectionRowData refreshWithSection:tableView:tableViewRowData:] + 2510
    6   UIKit                               0x00d20b3d -[UITableViewRowData numberOfRows] + 98
    7   UIKit                               0x00ba94c2 -[UITableView noteNumberOfRowsChanged] + 120
    8   UIKit                               0x00ba8e6f -[UITableView reloadData] + 814
    9   UIKit                               0x10378ed1 -[UITableViewAccessibility(Accessibility) reloadData] + 50
    10  UIKit                               0x00bac8d3 -[UITableView _reloadDataIfNeeded] + 65

它提供了更多,但似乎都源于我发布的第一个错误。如果它们有帮助,请告诉我,我会发布它们。

不过,这个问题真的很奇怪。它似乎不知从何而来。即使我使用 git 恢复到以前的工作版本,它仍然会崩溃,所以我不知道是什么原因造成的。

Stackoverflow上发布了一个解决方案,但它有点令人不安。将缓存设置为 nil 是否会从我的应用程序中移除缓存功能/使其变慢?它会带来什么后果?

我正在对我的应用程序进行新的更新,首要任务是确保我的现有/未来用户不会崩溃。


编辑:我相信这就是导致问题的原因(尽管我已经删除了它并且问题仍然存在):

我在 AppDelegate 的applicationDidFinishLaunching中有以下代码,它基本上用于在应用程序首次启动时在核心数据中创建一个特殊对象。添加此内容后,尽管此后不久删除了代码,但问题似乎发生了。我不应该在这一点上操纵核心数据(现在还为时过早)吗?

// If it's the first time launching, create and add a sample article for an introduction
if ([[[NSUserDefaults standardUserDefaults] objectForKey:@"IsFirstTimeLaunching"] isEqualToString:@"YES"]) {
    NSManagedObjectContext *context = self.managedObjectContext;
    Article *article = [NSEntityDescription insertNewObjectForEntityForName:@"Article" inManagedObjectContext:context];
    article.source = @"text";
    article.body = @"some text";
    article.timeStamp = [NSDate date];
    NSError *error;
    [context save:&error];
    [[NSUserDefaults standardUserDefaults] setObject:@"NO" forKey:@"IsFirstTimeLaunching"];
}

我的AppDelegate中也有这个块,它已经存在了很长时间,但老实说,我不确定它到底做了什么,我可能刚刚写了它而忘记完成它......

- (void)applicationDidEnterBackground:(UIApplication *)application
{
    NSManagedObjectContext *context = self.managedObjectContext;
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Article" inManagedObjectContext:context];
    NSFetchRequest *request = [[NSFetchRequest alloc] init];
    [request setEntity:entity];    
}

您有以下选项:

  1. 将缓存设置为 nil 。如果您没有遇到性能问题,这实际上是可以的。如果获取需要很长时间,FRC 缓存确实可以提高速度。首先尝试此解决方案。

  2. 设计具有不同缓存名称的方案。FRC 可以确定在延迟实例化期间必须执行的提取,并使用适当的缓存名称。例如,如果您有搜索控制器,则可以使用范围按钮的状态。
    对于此解决方案(如果您只使用一个缓存名称),如果您更改了 FRC 的fetchRequest,则不要在获取的结果控制器上调用 performFetch,这一点很重要。这正是导致这次崩溃的原因。相反,您可以将 FRC 设置为 nil 并让它懒惰地创建自己。

  3. 在适当的时间删除带有deleteCacheWithName:的缓存。

最新更新