在之前的一篇 博文 里我们给出了两种解决办法:
- 第一种方法预先不指定context,在实际插入时在指定
- 第二种方法是在验证失败时重置context中的内容
其实CoreData自身还提供了另一种方法,就是context子对象.
对于一个context来说,默认其parent属性指向永久存储协调器(NSPersistentStoreCoordinator),不过我们可以让其指向另一个context,这样做的好处是sub context的所有修改要想应用到底层数据库中必须起parent也进行save才可以.如果其parent context没有进行save,则内容会被全部丢弃,不会对底层造成任何影响.下面我们来看看如何利用这个特性来解决题目中的问题.
首先是创建sub context:
let childCtx = NSManagedObjectContext(concurrencyType: .mainQueueConcurrencyType)
childCtx.parent = containerVC.managedObjectContext
然后在创建新托管对象时使用childCtx:
if let entity = NSEntityDescription.entity(forEntityName: "KsMain", in: childCtx){
let ks = KsMain(entity:entity,insertInto:childCtx)
//fill ks's properties
在最终需要保存的时候调用如下代码:
func insertKsMain(ks:KsMain){
do{
try ks.validateForInsert()
try? ks.managedObjectContext?.save()
saveContext()
}catch{
print(error.localizedDescription)
}
}
注意以上代码的do块中需要save两次,第一次是childCtx的保存,后一次是parent context的保存.保存是递进型的,childCtx保存到parent Context中,而parent Ctx最终保存到底层数据库中.省略其中任何一个save,App表现的都不正确.而且如果在do块中出错,则childCtx的修改直接被丢弃,从而不会影响parent Ctx的内容,更别说底层数据库了 ;)
作者:mydo 发表于2017/1/6 14:13:44 原文链接
阅读:74 评论:0 查看评论