使用一对多关系向 CoreData 添加新对象

Posted

技术标签:

【中文标题】使用一对多关系向 CoreData 添加新对象【英文标题】:Adding new object to CoreData with one to many relation 【发布时间】:2016-01-13 23:43:42 【问题描述】:

我有这样的 CoreData 模型:

包裹只能有一个公司,但公司可以有多个包裹要投递。

我在数据库中预装了三个公司。我创建了带有部分的表格视图,并通过 NSFetchedResultsController 加载数据。 我是这样配置的:

let fetchRequest = NSFetchRequest(entityName: EnityNames.PackageInfoEnityName)

        // Add Sort Descriptors
        let sortDescriptor = NSSortDescriptor(key: PackageInfoKeyPaths.Company, ascending: true)
        fetchRequest.sortDescriptors = [sortDescriptor]

        // Initialize Fetched Results Controller
        let fetchedResultsController = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: self.managedObjectContext, sectionNameKeyPath: PackageInfoKeyPaths.Company, cacheName: nil)

        // Configure Fetched Results Controller
        fetchedResultsController.delegate = self

        return fetchedResultsController

当我在模拟器上启动应用程序时,我有三个部分(如预期的那样),我希望将部分显示为公司,所以我使用关系作为部分名称键路径:

number of sections 3
FedEx
UPS
DHL

我创建了弹出框,我可以在其中向数据库添加新条目以使用其他数据填充列表。这是我用来创建和保存新数据的代码:

let appDelegate = UIApplication.sharedApplication().delegate as! AppDelegate
        let package = 

    NSEntityDescription.insertNewObjectForEntityForName(EnityNames.PackageInfoEnityName, inManagedObjectContext: managedObjectContext) as! PackageInfo
            let formatter = NSNumberFormatter()
            formatter.numberStyle = .DecimalStyle
            package.parcelNumber = formatter.numberFromString(parcelNumberTextField.text!) ?? 0;
            package.createdAt = NSDate()
            let company = avaialbleCompanies![companyPickerView.selectedRowInComponent(0)]
            package.company_relation = company
            company.addPackageToCompany(package)    


            appDelegate.saveContext()

company 数组是在为我的小弹出窗口做准备时传递的,让用户只选择数据库内的公司。在这之后我得到了一些奇怪的东西:

number of sections 4
FedEx
UPS
UPS

为什么要添加新的部分?它应该只是将新项目添加到现有部分!

还有可怕的错误:

PackageChecker[5336:281349] CoreData:错误:严重的应用程序错误。在调用 -controllerDidChangeContent: 期间,从 NSFetchedResultsController 的委托中捕获了异常。无效更新:无效的节数。更新后表视图中包含的节数(4)必须等于更新前表视图中包含的节数(3),加上或减去插入或删除的节数(0插入,0已删除)。与 userInfo (null)

重要的部分可能是我得到了这个:

updated index path 1x0

我正在使用fetchedResultsController.sections?.count 来获取部分的数量。 如果我在 sim 上重新启动应用程序,我新添加的记录会出现在列表内的适当部分中。为什么它在运行时没有正确更新?

附:我有:

func controllerWillChangeContent(controller: NSFetchedResultsController) 
        self.tableView.beginUpdates()
    

    func controllerDidChangeContent(controller: NSFetchedResultsController) 
        self.tableView.endUpdates()
    

编辑:

如果我删除 sectionNameKeyPath: PackageInfoKeyPaths.Company - 添加效果很好。你能帮我做这部分吗?也许我配置不好。

【问题讨论】:

您是否实现了其他NSFetchedResultsControllerDelegate 方法controller:didChangeObject:atIndexPath:forChangeType:newIndexPath:controller:didChangeSection:atIndex:forChangeType: @pbasdf 我添加了 ontroller:didChangeSection:atIndex:forChangeType: 因为它丢失了。现在列表有效,我可以添加项目而不会出错!太棒了。但是,仍然将新项目添加到现有部分,而是在添加后创建新部分。当我杀死我的应用程序并打开它时,我在适当的部分获得了新添加的项目,我在插入时做错了什么,它正在向现有的部分添加新部分而不是新元素? 唯一看起来不寻常的行是:company.addPackageToCompany(package) 如果addPackageToCompany 照它说的做,那是多余的:当执行package.company_relation = company 时,CoreData 会自动设置反比关系。 @pbasdf 我后来加了,没了也一样。我应该使用与获取结果中使用的相同的 managedObjectContext 吗?我应该创建一个新的吗?我正在使用 AppDelegate 中的 managedObjectContext。 是的,当您设置关系时,两个对象必须在同一个上下文中。 PackageInfoKeyPaths.Company 的值是多少? 【参考方案1】:

发生崩溃是因为 FRC 正在创建一个新部分,但您的代码当前没有创建相应的 tableView 部分。如果你实现NSFetchedResultsControllerDelegate 方法:

controller:didChangeSection:atIndex:forChangeType:

这应该可以解决崩溃问题。

但恐怕我看不出 FRC 为什么要创建一个新部分。

【讨论】:

以上是关于使用一对多关系向 CoreData 添加新对象的主要内容,如果未能解决你的问题,请参考以下文章

Restkit + Coredata - 仅将唯一对象添加到一对多关系中

NSPredicate:“添加”一对多关系CoreData中所有实体的属性值

获取对象时无法访问一对多CoreData关系

Core Data 使用多个上下文中的新对象从后台线程订购一对多关系保存

CoreData 向实体行添加新关系

以一对多关系从 coredata 中获取所有对象