更改无法识别我的 NSFetchedResultsController 附加到后台线程

Posted

技术标签:

【中文标题】更改无法识别我的 NSFetchedResultsController 附加到后台线程【英文标题】:Changes not recognized my NSFetchedResultsController attached to background thread 【发布时间】:2014-12-17 02:24:07 【问题描述】:

我正在使用 MagicalRecord,但不确定这是 MagicalRecord 还是 CoreData 中的错误。

当用户单击新按钮并将其保存到默认上下文时,我会在 UI 线程中创建新实体:

- (IBAction) saveToCoreData:(UIButton *)sender 
    NSManagedObjectContext *context = [NSManagedObjectContext MR_defaultContext];
    Report *report = [Report MR_createInContext:context];
    report.dirty = [NSNumber numberWithBool:YES];
    report.timestamp = [NSDate date];

    Document *document = [Document MR_createInContext:context];
    document.dirty = [NSNumber numberWithBool:YES];
    document.timestamp = [NSDate date];
    document.report = report;

    NSLog(@"Saving.................................");
    [context MR_saveToPersistentStoreWithCompletion:^(BOOL success, NSError *error) 
        NSLog(@"Saved report and document to PS");
    ];

请注意,我有一个附有子文档的报告实体

然后我有一个后台线程只负责监听变化,如果报告发生变化,dirty == YES 那么它需要提交到服务器。

我在后台线程上设置了 NSFetchedResultsController 来监听更改,以便知道何时保存到服务器。

// This should get me a new background context, use it in FRC below
    self.fetchedResultsController = [Report MR_fetchAllSortedBy:@"timestamp"
                                                      ascending:NO
                                                  withPredicate:[NSPredicate predicateWithFormat:@"dirty == YES"]
                                                        groupBy:nil
                                                       delegate:self
                                                      inContext:_managedObjectContext];

然后我实现监听变化的方法:

- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id) anObject atIndexPath:(NSIndexPath *) indexPath forChangeType:(NSFetchedResultsChangeType)type newIndexPath:(NSIndexPath *) newIndexPath 
    switch(type) 

        case NSFetchedResultsChangeInsert:
            NSLog(@"report inserted");
            [self pushReport:anObject];
            break;
        case NSFetchedResultsChangeUpdate:
            NSLog(@"report updated");
            [self pushReport:anObject];
            break;
    

并将其推送到服务器:

- (void) pushReport:(Report *) report 
    __weak ReportPushService *weakSelf = self;
    AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
    [manager GET:@"http://echo.jsontest.com/" parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) 
        [weakSelf.managedObjectContext performBlockAndWait:^
            report.remoteId = @"12345"; // fake id from server
            report.dirty = [NSNumber numberWithBool:NO];

            [weakSelf.managedObjectContext MR_saveToPersistentStoreWithCompletion:^(BOOL success, NSError *error) 
                NSLog(@"Saved server info to Report");
            ];
        ];
      failure:^(AFHTTPRequestOperation *operation, NSError *error) 
         NSLog(@"Error: %@", error);
     ];

FRC 的工作原理就像一个魅力,当 UI 线程添加一个带有dirty == YES 的新报告时,FRC 会拾取它并将其提交给服务器。

这是我遇到问题的地方。将报告保存到服务器后,服务器会返回该报告的 ID。我将其设置为 report.remoteId。

我在另一个类中有一个 FRC,负责将文档保存到服务器。它将检查文档脏 == YES 并且它的父(报告)remoteId != nil,即:

    self.fetchedResultsController = [Document MR_fetchAllSortedBy:@"timestamp"
                                                      ascending:NO
                                                  withPredicate:[NSPredicate predicateWithFormat:@"report.remoteId != nil && dirty == YES"]
                                                        groupBy:nil
                                                       delegate:self
                                                      inContext:_managedObjectContext];

它使用与报告“推送”类中的 FRC 控制器相同的背景 MOC。

然后我监听该 FRC 的变化:

- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id) anObject atIndexPath:(NSIndexPath *) indexPath forChangeType:(NSFetchedResultsChangeType)type newIndexPath:(NSIndexPath *) newIndexPath 
    switch(type) 
        case NSFetchedResultsChangeInsert:
            NSLog(@"document inserted");
            [self pushDocument:anObject];
            break;
        case NSFetchedResultsChangeUpdate:
            NSLog(@"document updated");
            [self pushDocument:anObject];
            break;
    

但是他的 FRC 从来没有看到任何变化,即使我已经确认我有一个 remoteId != nil 的报告和一个脏 == YES 的(子)文档。

为什么 Documents FRC 看不到与该谓词匹配的对象?

【问题讨论】:

你是否在 UI 线程中观察到 MOC 的 NSManagedObjectContextDidSaveNotification 并在后台线程中将更改合并到 MOC 中? 您需要描述如何设置 MOC 以及如何管理线程(您是否始终在 MOC 的正确线程上,并且更改是否在您的 MOC 中合并) 行为不端的 NSFetchedResultsController 是否曾经执行过抓取?它在正确的上下文中吗?是否实现了所有必需的委托方法和 NSFetchedResultsController 的委托集? 【参考方案1】:

FRC 的问题/限制是它只包含为其设置的实体。在这种情况下:

self.fetchedResultsController = [Document MR_fetchAllSortedBy:@"timestamp"
                                                      ascending:NO
                                                  withPredicate:[NSPredicate predicateWithFormat:@"report.remoteId != nil && dirty == YES"]
                                                        groupBy:nil
                                                       delegate:self
                                                      inContext:_managedObjectContext];

FRC 将仅包含文档实体。初始提取将作用于报告关系。但如果报告发生变化,即使谓词匹配,FRC 也不会更新。

http://www.mlsite.net/blog/?p=825

这里

Changing a managed object property doesn't trigger NSFetchedResultsController to update the table view

【讨论】:

以上是关于更改无法识别我的 NSFetchedResultsController 附加到后台线程的主要内容,如果未能解决你的问题,请参考以下文章

更改无法识别我的 NSFetchedResultsController 附加到后台线程

为啥 git 无法识别我的文件已更改,因此 git add 不起作用

Visual Studio 无法识别环境变量的更改

核心数据难题:不能两次调用 NSFetchedResults

我所在国家/地区的夏令时已更改,但 C#/Windows 无法识别更改

SSL:目的地更改 IP 地址后无法识别证书