Core Data 是苹果为OS X 和iOS系统应用开发提供的数据持久化技术。它基于高级数据持久化的API,它的底层最终是SQLite数据库、二进制文件和内存数据保存,这样开发人员不用关心数据的存储细节问题,不用再使用SQL语句,不用面对SQLite的C语言函数
一、前言
因为关于Core Data的文章多如牛毛,一些太偏概念性的东西不再介绍。本文只陈述Core Data的使用过程和细节详解。配带简单的Demo帮助理解
虽然创建项目的时候,选择使用User Core Data,会自动添加一些代码。但这些代码都是基于最新操作系统的,比如,我现在使用的是Xcode 8.2.1,创建的代码是基于iOS 10以上的系统的代码。本文的操作是基于iOS 8.0以上的系统的。并且多数据结构做了封装。本文直接对一个小型项目(日记本)的操作过程做详解。以下开始我们的项目创建过程
二、项目启动
1、创建Data Model文件
创建数据库文件2、添加表和属性
表名和属性3、通过模型创建类文件
弹出的对话框,选择对应的模型文件和表文件,下一步就可以了,会生成两个文件生成的这两个文件,在我们的使用过程中,没什么用~,但是删除后,会有提示,模型没有对应的类描述,使用默认的类,当然运行结果是正常的,这个提示挺烦的,我就加上了,并且单独用一个文件夹放起来了。
4、需要试图完成的过程已经结束了,接下来我们分析Core Data的堆栈
被管理的上下文<—>持久化存储协调器<—>持久化对象存储<—>存储文件,另外 持久化存储协调器<—>被管理的对象模型。他们之间存在这样的关系。而我们需要对这一部分做封装,因为每次的数据存储都会经历这个过程。我们定义为(CoreDataBase)
先设定一下我们的代码结构。CoreDataBase(封装好的数据库存储基类),NoteCore(实体操作类),NoteManagedObject(管理的实体类),Note(模型类)。NoteViewModel(视图模型类)
CoreDataBase的封装,就针对于上述的对象,做一些处理
lazy var applicationDocumentsDirectory : URL = {
let urls = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)
return urls[urls.count - 1]
}()
// MARK: - 被管理的对象上下文
lazy var managedObjectContext : NSManagedObjectContext? = {
let coordinator = self.persistentStoreCoordinator
if coordinator == nil {
return nil
}
var managedObjectContext = NSManagedObjectContext()
managedObjectContext.persistentStoreCoordinator = coordinator
return managedObjectContext
}()
// MARK: - 返回持久化存储协调者
lazy var persistentStoreCoordinator : NSPersistentStoreCoordinator? = {
var coordinator : NSPersistentStoreCoordinator? = NSPersistentStoreCoordinator(managedObjectModel: self.managedObjectModel)
let url = self.applicationDocumentsDirectory.appendingPathComponent("note.sqlite")
var error : NSError? = nil
do {
try coordinator?.addPersistentStore(ofType: NSSQLiteStoreType, configurationName: nil, at: url, options: nil)
}catch{}
return coordinator
}()
// MARK: - 返回被管理的对象模型
lazy var managedObjectModel : NSManagedObjectModel = {
let modelURL = Bundle.main.url(forResource: "note", withExtension: "momd")!
return NSManagedObjectModel(contentsOf : modelURL)!
}()
Note(模型类)
var title : String?
var detail : String?
var time : Data?
init(dict : [String : AnyObject]){
super.init()
setValuesForKeys(dict)
}
override func setValue(_ value: Any?, forKey key: String) {
super.setValue(value, forKey: key)
}
override func setValue(_ value: Any?, forUndefinedKey key: String) {}
NoteCore(实体操作类)
我们先写插入的方法,测试功能是否正常
func create(model : EilNote){
let cxt = self.managedObjectContext!
let note = NSEntityDescription.insertNewObject(forEntityName: "Note", into: cxt)
note.setValue(model.title!, forKey: "title")
note.setValue(model.detail!, forKey: "detail")
note.setValue(model.time!, forKey: "time")
do {
try cxt.save()
print("插入成功")
}catch{
print("error")
}
}
测试代码
let dict : [String : Any] = [
"title" : "标题",
"detail" : "今天天气不错,挺风和日丽的",
"time" : NSDate()
]
let note = EilNote(dict: dict as [String : AnyObject])
NoteCore().create(model: note)
输出插入成功,应该是没问题了,一会我们写查找功能。在此之前我先说明一下,运行成功前,我代码崩溃了好几次。有这么几种情况
a、类型名称不匹配,模型的类型和表中的字段一定要一致
b、如果出现addPersistentStoreWithType:SQLite configuration:(null),类似这样的警告描述,不用害怕,删除App,重新执行一下就好了,可能是因为已经存在了错误的配置
c、返回被管理的对象模型,里边的文件名称是你的Data model的名称,在你的项目中,假设是NoteData.xcdatamodeld的名称,你的文件名就是 NoteData,类型是momd。而返回持久化存储协调者,里边这个名称是可以自己命名的,我看到有人的描述都用同一个名称。当然即便是使用同一个名称也没有什么问题。
d、实体名称是你的表名。
如果一直运行报错,一定删除App,重新编译。
我们补全NoteCore查找方法
func findAll() -> NSMutableArray { let cxt = self.managedObjectContext! let entity = NSEntityDescription.entity(forEntityName: "Note", in: cxt) let fetchRequest = NSFetchRequest()
fetchRequest.entity = entity
//筛选的条件
let sortDescriptor = NSSortDescriptor(key: "title", ascending: true)
fetchRequest.sortDescriptors = [sortDescriptor]
do {
let listData = try? cxt.fetch(fetchRequest)
let resListData = NSMutableArray()
for item in listData! {
let mo = item as! NSManagedObject
let title = mo.value(forKey: "title") as AnyObject
let detail = mo.value(forKey: "detail") as AnyObject
let time = mo.value(forKey: "time") as AnyObject
let dict = [
"title" : title ,
"detail" : detail ,
"time":time
]
let note = EilNote(dict: dict)
resListData.add(note)
}
return resListData
}
}
测试代码
print(NoteCore().findAll())
打印了我们刚才插入的数据,接下来,我们完善note这个功能。并且添加,修改和删除的方法。
三、demo还原场景
下边是demo的链接
网友评论