Constrain
有时我们需要保持一个字段的唯一性,比如微信号——因为重复会让事情变的很麻烦。为达到这个目的,我们申明Attribute为“Constrained”,以保证它的唯一性。
设立
选中Entity,并通过菜单->View->Inspector->显示Data Model Inspector窗口(或者直接打开Xcode右上角那个“显示检查器“Hide or Show Inspectors””)的东东。
然后就能找到Constraints了。
我们点击加号“+”添加一个Constraint,按回车重命名,把它的名字改成我们想要constrain的Attribute的名字。
例如,我想让上一章的“书籍”程序中的书名“title”是唯一的,于是我们加上title。
Core Data 大!注!意!
干啥事儿之后最好都Cmd+S储存你的改动!因为他很呆!不一定会自动识别出来!还有可能给你搞编译错误!
⬇️没保存的后果,编译不通过并且能运行
为了方便演示“constrain”的功能,我们新建一个项目,往其中添加“Student”的entity和“name”的String的attribute,并把“name“设为constrain。
【记得Cmd+S】
用一个List来显示students的所有名字,并用一个“Add”添加学生,“Save”保存到数据库;
struct ContentView: View {
@FetchRequest(entity: Student.entity(), sortDescriptors: []) var students: FetchedResults<Student>
// 读取Student的entity
@Environment(\.managedObjectContext) var moc
// 修改Student的entity中的context
var body: some View {
VStack{
List(students, id:\.self){ student in
Text(student.name ?? "Unknown")
} // List所有学生
Button("ADD") { //添加学生
let NewStudent = Student(context: self.moc)
NewStudent.name = "QSD"
}
Button("SAVE"){ //尝试保存学生
do{
try self.moc.save()
} catch {
print(error.localizedDescription) //若不保存成功,则print错误
}
}
}
}
}
运行后我们发现,当添加多个QSD并准备保存时,系统会提示:
The operation couldn’t be completed. (Cocoa error 133021.)
让Core Data写下名字
如果你执意要让Core Data写下至少一个名字,可以在SceneDelegate.swift
开头加上Import CoreData
,然后,在
let context = (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext
后一行加入:
context.mergePolicy = NSMergeByPropertyObjectTrumpMergePolicy
这段代码让Core Data整合(Merge)冲突的多个数据,用新的覆盖旧的。
再次运行,我们发现添加多个QSD并尝试保存后,QSD会变成一个。
为了验证“用新的覆盖旧的”理论,我们回到我们的“书籍”App中,将“title”设为constrain,在“Add Book”按钮上加入类似的代码,并也在SceneDelegate.swift
中加入类似声明,执行完毕后我们尝试加入新的title一样的书,你会发现这本新书的内容完全覆盖了老的。