美文网首页
swift3.0 之Core Data

swift3.0 之Core Data

作者: Eil_tea | 来源:发表于2016-12-22 17:42 被阅读114次

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的链接

相关文章

网友评论

      本文标题:swift3.0 之Core Data

      本文链接:https://www.haomeiwen.com/subject/jrmcvttx.html