使用一对多关系向 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中所有实体的属性值