【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记46 Persistence持久化

本话将介绍IOS中的四种数据持久化方式:

Archiving

SQLite

File System

Core Data

前面我们将结果NSUsrDefaults的用法,它是针对小数据量的持久化技术,本话的四种方式是针对大数据量的操作。

1.Archiving

Archiving是一种把对象存储到硬盘上的存储方式,被存储的对象不需要所有属性的目录。只要一个对象实现了Archiving的两个方法,那么使用Archiving就可以存储这个对象的所有分支属性:

func encodeWithCoder(encoder:NSCoder)
init(coder:NSCoder)

encoderWithCoder这个方法就是把某个对象以字典的形式写出来,然后再进行编码。当这些对象需要被读取时,init就会初始化它们。storyboard就是使用这种技术存储的,所有的UIView、UIViewController都是可以编码的,因为它们都遵守了NSCoding协议。比如当某个UIView被保存时,其包含的所有对象所组成的图表都会被保存下来。所以当我们使用init创建一个UIViewController的时候会需要一个init Coder的属性。这种机制非常适合storyboard,但是不推荐我们使用。

2.SQLite

SQLlte是一个移动端的SQL数据库,操作SQL数据库的很多API是C风格的,并不适合Swift程序员。

3.File System

只要是使用IOS的设备,都会有一个基于Unix内核的文件系统。使用File System保存数据就是存储到这个文件系统中,你需要分四步来实现这个操作:

(1)指定根路径,比如Documents目录、Caches目录

(2)将文件目录增加到URL中,使用append方法

(3)然后就可以根据这个目录读写文件了

(4)管理文件系统,使用NSFileManager

文件系统的URL都是从“/.”开始的,文件系统会为每个应用单独开辟存储空间,不同应用间的这部分存储空间是隔离的,这就是所谓的“沙盒”。即便是在沙盒里面,有些部分也是不允许写入的,比如存放应用自身的地方。你可以读取沙盒中的所有部分。

至于为什么使用“沙盒”,首先是出于安全性的考虑。其次在你删除某个应用的时候,会把它的沙盒一同删除。备份也是一样,备份特定沙盒中的内容,而不是所有的目录。

沙盒中有各种bundles,以目录的形式存在。

你的应用占用的是Application bundle 目录,里面会保存所有与应用相关的内容,比如storyboard、图片等等,它们都是不可写的,如果要修改你需要把它们取出来放到Application bundle目录之外的目录中进行修改

Documents 目录存放用户数据。

Caches 目录存放那些可以被丢弃的数据。

那么你如何能得到一个访问这些目录的路径呢?使用NSFileManager:

let fileManager = NSFileManager()

NSFileManager是线程安全的,你只能在创建NSFileManager实例的线程中调用这个实例。DefaultFileManager是在主线程中调用的,否则就是新建一个实例,创建NSFileManager的实例是轻量级的。有了一个FileManager之后就可以访问它。这个方法返回的是一个数组,这是由于在Mac中调用这个API可能会返回很多URL,但是在iphone上你只能获得用户的资料,我们只需要数组中的第一个元素就可以了:

let urls:[NSURL] = fileManager.URLsForDirectory(NSSearchPathDirectory,inDomain:NSUserDomainMask)

NSSearchPathDirectory参数根据需要选择你想要获取的目录,比如传入NSDocumentsDirectory。你可以在这些路径中写入一些新东西,使用NSURL的方法:

URLByAppendingPathComponent(NSURL)和

URLByAppendingPathExtension(NSURL)

有时你得到一个URL之后想知道这是不是某个URL,比如某个文件的URL,系统为我们提供了相应的方法:

isFileReferenceURL()

还有一个重要的方法:

public func resourceValuesForKeys(keys: [String]) throws -> [String : AnyObject]

使用这个方法传递你想要知道的信息的key到keys参数中,可以查询该文件的最后访问时间、尺寸等等信息,这些信息会在返回的字典中返回。

想要向URL中存入文件的话使用NSData,声音、图像等通常都会以NSData的格式存储:

public func writeToURL(url: NSURL, atomically: Bool) -> Bool 

atomically参数的意思是先写一个临时文件,然后删掉老的文件,再把新的文件放进去,保持原子性。

4.Core Data

如果你了解了Core Data,你就了解了更多IOS工作的原理。

Core Data会使用面向对象的方式去向SQL中存取数据,而不用使用SQL命令。使用Core Data时你需要预先设置好数据库中的表和对象的映射,数据库中模型的每一个行对应对象中的每个属性,属性要使用

@NSManaged var来创建,代表着你在更改属性的值的时候会同时改变数据库中的值。你可以用可视化的方式创建模型。

然后拿到数据库的handle(控制器)或者叫hook(钩子):

UIManagedObjectContext,这个变量需要定义在AppDelegate中。另外在创建工程的时候选择“Use Core Data”。另外还有一个类UIManagedDocument,它使对于一个Core Data数据库的封装。UIManagedDocument最酷的一点是,如果你使用它,iCloud部分会简单很多。使用context,你可以创建数据库中的对象,使用NSEntityDescription类的insertNewObjectForEntityForName

还有 executeFetchRequest方法,传入NSFetchRequest类型的参数,可以设置查询请求去查询数据库中的数据,得到NSArray类型的返回值,成员类型是NSManagedObject。这就是你对数据库可以进行的操作。一旦你持有这些NSManagedObject对象,你可以使用这两个方法:setValue和valueForKey来操作对象,或者创建NSObject的子类,通常来说子类和数据库中的实体同名:

class Photo:NSObject {
@NSManaged var title:String
}

然后使用set和get方法操作属性:

let photo = NSEntityDescription.insertNewObjectForEntityForName("Photo",inMain...)
photo.title = "My First Photo"

Core Data同样擅长跟Table View配合,因为如果Core Data中有很多数据的情况下,你可能需要使用Table View去显示。在IOS中有一个类NSFetchedResultsController的类,你给它一个查询请求,它会关联一个TableView,这个TableView和你的数据库是实时关联的,并且这个类会实现TableView的所有datasource方法。

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-08 00:10:13

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记46 Persistence持久化的相关文章

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记1 IOS8概述

首先感谢网易公开课和SwiftV课堂的朋友们辛苦翻译,这个系列是我学习斯坦福IOS8公开课的个人心得体会和笔记,希望能给大家带来启发. 首先我们要知道IOS系统中的结构情况,从贴近硬件的底层到贴近用户的顶层,分为四个层次: 1.Core OS层在最下层,很多人可能不知道IOS是一个基于UNIX的操作系统,它大量借鉴了Mac os X 的内核部分,Mac OS X我们肯定不会陌生,命令行的使用很好的证明了它是一个基于UNIX的系统.IOS针对移动设备对电池等硬件进行了系统的优化,但它仍可被看成是一

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记2 Xcode、Auto Layout及MVC

原文链接不知道在哪, 接着上一话来讲,上一话中讲到了MVC,那么MVC在IOS8开发中是如何应用的呢?Paul Hegarty老师给我们展示了一个计算器的Demo,首先新建一个工程,老师把AppDelegate.swift.LaunchScreen.xib和Images.xcassests文件放到了supporting Files文件夹中,那么剩下的两个文件ViewController.swift就是MVC中的C(控制器),Main.storyboard就是MVC中的V(视图). 在Main.s

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记13 Drawing绘制、UIColor颜色、Fonts字体

上一话介绍了视图绘制的一些基本原理,这一话继续展开.UIBezierPath可以绘制许多有趣的图形. 使用不同的构造器,比如roundedRect就是四个角被磨圆了的矩形,或者干脆是椭圆和圆.你甚至可以剪切任意的path,剪切使用addClip方法,在剪切了之后你可以针对剪切的这部分进行操作,例如你正在绘制一个卡片,这个卡片有小小的圆角效果,你可以把卡片绘制在一个矩形里面,然后把它剪切到一个小一点的圆角矩阵里,这样四角就修圆了.这是一个获得圆角卡片最简单的方法. 你也可以进行碰撞检测,它基于绕圈

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记37 TableView Delegate

上一话介绍了tableView的datasource,本话来介绍另一个重要的部分delegate. 当我们点击一个cell的时候,如何跳转到另外一个mvc中呢? 像增加其他segue一样,点击cell按住control键,右键连线到另一个mvc上,然后松手,选择需要的segue类型. 如果你的cell上还有其他按钮,比如detail disclosure,你也可以选择它的segue: 然后设置你的segue: 接着去prepareForSegue中设置这个segue: 每一个case对应不同的i

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记3 Xcode、Auto Layout及MVC

继续上一话中的计算器Demo.上一话讲到类必须被初始化.类中的属性也必须被初始化,所以你不能仅仅声明而不给它一个处置,那么问题来了,我们从storyboard中拖拽的@IBOutlet为什么仅仅有声明而不须要初始化呢,这是由于它的类型依然是一个optional,在你初始化之前已经被赋值为nil了,这也就是为什么你不须要再初始化它的原因. @IBOutlet weak var display: UILabel! 然而既然它是一个optional类型的,那为什么UILabel后面是"!"而

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记39 Alert&ActionSheet

Alert和ActionSheet是IOS中弹出消息的两个工具. 首先它们都是Modal的方式展示的. Alert用来向用户发起询问,可以有一个(比如取消)或两个选项(比如确定和取消),也可以附带一个文本框(比如要求用户输入密码) Action Sheet从屏幕底部滑出,提供一些分支的选项,选项的数量可以大于两个. 对比如图: Action Sheet和Alert都可以使用UIAlertController来创建,比如创建一个Action Sheet,在构造器中制定它的title和简介,注意St

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记23 多MVC模式Demo的实现

上一话我们对Demo的选择界面做了自动布局的相关处理,现在开始连接多个MVC的操作.首先我们需要其他工程中的文件,那么让我们打开另一个app.点击下面这个文件 然后拖动我们需要的文件到新的工程目录下: 注意勾选第一行,不然只是做了引用,如果你不小心删除了目标目录的话,你就找不到这些文件了,所以还是推荐做复制,这样会把文件复制到我们自己的工程目录下. 那么storyboard中的内容怎么办呢,我们只需要在原工程的storyboard中选中控制器然后复制,粘贴到新工程的storyboard中即可.

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记31 Multithreading多线程

在IOS中存在着许多队列,和我们数据结构中的队列一样,这里的队列概念也是先进先出的.而每一个方法(包括闭包)都被组织在这些不同的队列中,而每一个队列都有自己的线程去运行这些队列,这就造就了多线程环境. 其中有一个非常重要的队列叫做主队列,主队列是一个串行队列,所以主队列只会一个一个地执行主队列中的函数.所有的UI活动都必须发生在主队列中,所以当你想要一个函数或者是闭包的时候就会执行某些代码,这就会做任何关于UI的事,必须把它放到主队列中,这是保护UI的好办法,主队列绝对不想做任何可能被阻塞的事情

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记6 init

这一话首先来讲写关于init的东西. 首先初始化并不会经常被用到,这是因为类和结构体中的大部分属性都会通过赋值被初始化,或者有些属性是Optional的,这样即使是nil也没关系,可以在之后再给它们赋值,就好比StoryBoard中的outlet,又或者可以使用闭包来初始化,或者使用lazy来避开init,所以有很多方法来避免init,除非你确实需要一个init的时候,那么该怎么做呢? 在一些情况下会自动生成init,其中一种情况是当类中的所有属性都有初始值的时候,你会自动得到一个没有参数的初始