用于关系的嵌套 NSManagedObjectContext 和 NSFetchedResultsController

Posted

技术标签:

【中文标题】用于关系的嵌套 NSManagedObjectContext 和 NSFetchedResultsController【英文标题】:Nested NSManagedObjectContext's and NSFetchedResultsController for relationships 【发布时间】:2014-03-21 13:23:05 【问题描述】:

尝试使用 MagicalRecord 为我的 CoreData 实体实现添加/保存控制器,并使用按钮保存和取消。

- (void)addProduct

    NSManagedObjectContext *context = [NSManagedObjectContext MR_context];

    [context MR_setWorkingName:@"PRODUCT_ADD_MOC"];

    ProductBaseEntity *entity = [ProductBaseEntity MR_createInContext:context];
    [self presentProductSaveControllerWithEntity:entity andContext:context];


- (void)editProduct:(ProductBaseEntity *)entity

    [self presentProductSaveControllerWithEntity:entity andContext:nil];


- (void)presentProductSaveControllerWithEntity:(ProductBaseEntity *)entity
                                    andContext:(NSManagedObjectContext *)parentContext

    if (!parentContext) 
        parentContext = [NSManagedObjectContext MR_contextForCurrentThread];
    

    NSManagedObjectContext *context = [NSManagedObjectContext MR_contextWithParent:parentContext];
    [context MR_setWorkingName:@"PRODUCT_SAVE_MOC"];

    ProductBaseEntity *contextEntity = (ProductBaseEntity *)[context objectWithID:entity.objectID];

    ProductSaveController *controller = [[ProductSaveController alloc] initWithEntity:contextEntity];

    controller.managedObjectContext = context;

    controller.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithTitle:@"Cancel"
                                                                                   style:UIBarButtonItemStylePlain
                                                                                  target:controller
                                                                                  action:@selector(cancel)];

    controller.navigationItem.rightBarButtonItem = [[UIBarButtonItem alloc] initWithTitle:@"Save"
                                                                                    style:UIBarButtonItemStylePlain
                                                                                   target:controller
                                                                                   action:@selector(done)];

    [controller setDoneHandler:^(ProductSaveController *saveController) 
        if (context.hasChanges) 
            [context MR_saveToPersistentStoreWithCompletion:^(BOOL success, NSError *error) 
                [self dismissViewControllerAnimated:YES completion:nil];
            ];
         else 
            [self dismissViewControllerAnimated:YES completion:nil];
        
    ];

    [controller setCancelHandler:^(ProductSaveController *saveController) 
        if (context.hasChanges) 
            [OHAlertView showAlertWithTitle:@"Really exit?"
                                    message:@"Exit and discard changes?"
                               cancelButton:@"Cancel"
                                   okButton:@"Exit"
                             onButtonTapped:^(OHAlertView *alert, NSInteger buttonIndex) 
                                 if (buttonIndex == 1) 
                                     [self dismissViewControllerAnimated:YES completion:nil];
                                 
                             ];
         else 
            [self dismissViewControllerAnimated:YES completion:nil];
        
    ];

    [self presentViewController:[[UINavigationController alloc] initWithRootViewController:controller] animated:YES completion:nil];

问题的第一部分是:这个实现看起来不错吗?我为保存控制器创建单独的上下文,然后我可以决定:我应该将更改保存在持久存储中(保存按钮)还是应该在不修改持久存储的情况下丢弃更改(取消按钮)。对于 addProduct,我再创建一个上下文,因为如果我调用 context.hasChanges,它无论如何都会给我YES,因为对象已插入到此上下文中。

顺便说一句,我的这部分代码工作得很好。

ProductSaveController 中,我有一个打开SomeController 的按钮,其中包含与ProductBaseEntity 相关的实体(1:M)列表,称为ProductEntity,使用NSFetchedResultsController

_fetchedResultsController = [ProductEntity MR_fetchAllSortedBy:@"position"
                                                     ascending:YES
                                                 withPredicate:predicate
                                                       groupBy:nil
                                                      delegate:self
                                                     inContext:self.managedObjectContext];

其中self.managedObjetContext 是来自presentProductSaveControllerWithEntity 方法的上下文。问题是我的控制器没有反映这个控制器的变化。例如我这样做:

- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath

    if (editingStyle == UITableViewCellEditingStyleDelete) 
        ProductEntity *entity = [_fetchedResultsController objectAtIndexPath:indexPath];
        [entity MR_deleteInContext:_fetchedResultsController.managedObjectContext];
    

而我一无所有! tableView 中的这一行没有隐藏!但是,如果我单击后退按钮并再次打开此控制器,我会看到更改!如果我从 self.managedObjectContext 移动到 mainQueue 上下文,我的 fetchedResultsController 将按预期工作。 NSFetchedResultsController 是否应该与私有队列上下文一起使用??

【问题讨论】:

【参考方案1】:

您正在删除实体,但您似乎没有响应删除并更新您的视图。您通常可以通过在视图控制器上实现 NSFetchedResultsControllerDelegate 来做到这一点

【讨论】:

NSFetchedResultsController 工作,只要我使用 MR_defaultContext。似乎 NSFRC 在私有队列上下文方面存在问题。

以上是关于用于关系的嵌套 NSManagedObjectContext 和 NSFetchedResultsController的主要内容,如果未能解决你的问题,请参考以下文章

我需要知道用于查询嵌套关系并返回结果的 Laravel Eloquent 语法

如何更新脚手架生成的 MVC + 路由,用于在 Rails 中具有 2 个 belongs_to 关系的嵌套资源

Django Rest Framework 序列化器作为表单和嵌套关系

有两个循环,他们是嵌套关系,在内循环中使用break将终止哪一个循环?

用于迭代嵌套结果的 Django 查询集预取优化

SQL Server中的OpenJson用于嵌套的json数据?