保存和讀取數(shù)據(jù)的機(jī)制
在之前寫的Homeowner基礎(chǔ)上更新卧蜓,通過固化來保存和讀取模型對(duì)象,當(dāng)用戶重新運(yùn)行Homeowner應(yīng)用時(shí)把敞,可以讀取之前創(chuàng)建并保存的模型對(duì)象弥奸。
之前的應(yīng)用:《iOS編程(第四版)》Demo8:Homeowner
固化機(jī)制
固化 是由iOS SDK提供的一種保存和讀取對(duì)象的機(jī)制。當(dāng)應(yīng)用固化某個(gè)對(duì)象時(shí)奋早,會(huì)將該對(duì)象的所有屬性存入指定的文件夾盛霎。當(dāng)應(yīng)用解固某個(gè)對(duì)象時(shí),會(huì)從指定的文件讀取相應(yīng)的數(shù)據(jù)伸蚯,然后根據(jù)數(shù)據(jù)還原對(duì)象。
為了固化或解固某個(gè)對(duì)象简烤,相應(yīng)對(duì)象的類必須遵守 NSCoding 協(xié)議剂邮,并實(shí)現(xiàn)兩個(gè)必須方法:
@protocol NSCoding
- (void)encodeWithCoder:(NSCoder *)aCoder;
- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder; // NS_DESIGNATED_INITIALIZER
@end
- 在Item.m中實(shí)現(xiàn) NSCoding 協(xié)議:
固化:
- (void) encodeWithCoder:(NSCoder *)aCoder {
[aCoder encodeObject :self.itemName forKey :@"itemName"];
[aCoder encodeObject :self.serialNumber forKey :@"serialNumber"];
[aCoder encodeObject :self.dateCreated forKey :@"dateCreated"];
[aCoder encodeObject :self.itemKey forKey :@"itemKey"];
[aCoder encodeInt :self.valueInDollars forKey:@"valueInDollars"];
}
解固:
- (instancetype) initWithCoder:(NSCoder *)aDecoder {
self = [super init];
if (self) {
_itemName = [aDecoder decodeObjectForKey :@"itemName"];
_serialNumber = [aDecoder decodeObjectForKey :@"serialNumber"];
_dateCreated = [aDecoder decodeObjectForKey :@"dateCreated"];
_itemKey = [aDecoder decodeObjectForKey :@"itemKey"];
_valueInDollars = [aDecoder decodeIntForKey :@"valueInDollars"];
}
return self;
}
應(yīng)用沙盒機(jī)制
-
APP的沙盒文檔結(jié)構(gòu)
每個(gè)iOS應(yīng)用都有自己專屬的應(yīng)用沙盒(sandbox)。應(yīng)用沙盒就是文件系統(tǒng)中的目錄横侦,但是iOS系統(tǒng)會(huì)將每個(gè)應(yīng)用的沙盒目錄與文件系統(tǒng)的其他部分隔離挥萌。應(yīng)用只能訪問各自的沙盒。
應(yīng)用沙盒目錄
應(yīng)用程序包(application bundle)
包含應(yīng)用可執(zhí)行文件和所有資源文件枉侧,例如NIB文件和圖像文件引瀑。它是只讀目錄。Doucments/
存放應(yīng)用運(yùn)行時(shí)生成的并且需要保留的數(shù)據(jù)榨馁。iTune或iCloud會(huì)在同步設(shè)備時(shí)備份該目錄憨栽。當(dāng)設(shè)備發(fā)生故障時(shí),可以從iTunes或iCloud恢復(fù)該目錄中的文件翼虫。例如屑柔,Homepwner應(yīng)用可將用戶所擁有的物品信息保存在Documents/中Library/Caches/
存放應(yīng)用運(yùn)行時(shí)生成的需要保留的數(shù)據(jù)。與Documents/目錄不同的是珍剑,iTunes或iCloud不會(huì)在同步設(shè)備時(shí)備份該目錄掸宛。不備份緩存數(shù)據(jù)的主要原因是相關(guān)數(shù)據(jù)的體積可能會(huì)很大,從而延長(zhǎng)同步設(shè)備所需的時(shí)間招拙。如果數(shù)據(jù)源是在別處(例如web服務(wù)器)唧瘾,就可以將得到的數(shù)據(jù)保存在Library/Caches/目錄措译。當(dāng)用戶需要恢復(fù)設(shè)備時(shí),相關(guān)的應(yīng)用只需要從數(shù)據(jù)源(例如web服務(wù)器)再次獲取數(shù)據(jù)即可饰序。Library/Preferences/
存放所有的偏好設(shè)置领虹,iOS的設(shè)置(Setting)應(yīng)用也會(huì)在該目錄中查找應(yīng)用的設(shè)置信息。使用NSUserDefaults類菌羽,可以通過Library/Preferences/目錄中的某個(gè)特定文件以鍵值對(duì)形式保存數(shù)據(jù)掠械。iTunes或iCloud會(huì)在同步設(shè)備時(shí)備份該目錄。tmp/
存放應(yīng)用運(yùn)行時(shí)所需的臨時(shí)數(shù)據(jù)注祖。當(dāng)某個(gè)應(yīng)用沒有運(yùn)行時(shí)猾蒂,iOS系統(tǒng)可能會(huì)清除該應(yīng)用的 tmp/ 目錄下的文件,但是為了節(jié)約用戶設(shè)備空間是晨,不能依賴這種自動(dòng)清除機(jī)制肚菠,而是當(dāng)應(yīng)用不再需要使用 tmp/ 目錄中的文件時(shí),就及時(shí)手動(dòng)刪除這寫文件罩缴。iTune或iCloud不會(huì)在同步設(shè)備時(shí)備份 tmp/ 目錄蚊逢。通過NSTemporaryDirectory
函數(shù)可以得到應(yīng)用沙盒中的 tmp/ 目錄的全路徑。
獲取文件路徑
在實(shí)現(xiàn)保存和讀取模型對(duì)象的功能之前箫章,需要先獲取相應(yīng)文件的全路徑(Doucments/)
在HQLItemStore.m中編寫一個(gè)實(shí)現(xiàn)獲取路徑的方法:
- (NSString *) itemArchivePath {
//NSSearchPathForDirectoriesInDomains:獲取沙盒中某種目錄的全路徑
//注意第一個(gè)參數(shù)是NSDocumentDirectory而不是NSDocumentationDirectory
NSArray *documentDirectiorise = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
//從documentDirectiorise數(shù)組獲取第一個(gè)烙荷,也是唯一文檔目錄路徑
NSString *documentDirectiory = [documentDirectiorise firstObject];
return [documentDirectiory stringByAppendingPathComponent:@"items.archive"];
}
- 啰嗦一下:
這里是用的 HQLItemStore 倉庫類來統(tǒng)一管理 Item,因此這個(gè)獲取路徑的方法使用的是 實(shí)例方法檬寂,嗯终抽,并沒有任何問題。
試想桶至,如果有一個(gè)App需要保存用戶賬戶信息昼伴,而且一個(gè) App 就只有一條用戶信息,于是我們就把這個(gè)管理用戶賬戶的 Person 類設(shè)置為單例類镣屹,但是卻沒有必要?jiǎng)?chuàng)建倉庫類了圃郊,因?yàn)榫椭挥幸粭l用戶信息嘛,于是獲取沙盒路徑的方法也這樣寫女蜈?寫在 Person 類里面持舆?也寫成實(shí)例方法?可能就有點(diǎn)問題了伪窖。
// 單例類方法
+ (instancetype)sharedUser{
static Person *sharedUser = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// 根據(jù)路徑載入固化文件
NSString *path = [self itemArchivePath];
// ToDo
});
return sharedUser;
}
因?yàn)楫?dāng)從文件中解固之前要獲取對(duì)象吏廉,對(duì)象還沒有創(chuàng)建,先要得到路徑惰许,怎么能用實(shí)例化方法呢席覆?
- 解決方法是使用 C 函數(shù):
頭文件聲明:
#import <Foundation/Foundation.h>
// 聲明輔助函數(shù),用于返回文件路徑
NSString *docPath(void);
// ...
實(shí)現(xiàn)文件:
#import "Person.h"
// 輔助函數(shù)
NSString *docPath() {
NSArray *documentDirectories = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *documentDirectory = [documentDirectories firstObject];
return [documentDirectory stringByAppendingPathComponent:@"user.archive"];
}
調(diào)用:
Person *user = [NSKeyedUnarchiver unarchiveObjectWithFile:docPath()];
NSKeyedArchiver與NSKeyedUnarchiver
應(yīng)用退出前保存數(shù)據(jù)
Homepwner應(yīng)用在退出(exit)時(shí)汹买,通過NSKeyedArchiver類保存Item模型對(duì)象佩伤。
在HQLItemStore.h中聲明一個(gè)用于保存數(shù)據(jù)的新方法:
- (BOOL) saveChanges;
在HQLItemStore.m中實(shí)現(xiàn)該方法:
- (BOOL) saveChanges{
//獲取文件路徑
NSString *path = [self itemArchivePath];
//如果固話成功就返回YES
return [NSKeyedArchiver archiveRootObject:self.privateItems toFile:path];
}
- 在AppDelegate.m頂部導(dǎo)入HQLItemStore.h聊倔,然后實(shí)現(xiàn)
- (void)applicationDidEnterBackground:
方法,使用此方法可以釋放共享資源,保存用戶數(shù)據(jù)生巡,使計(jì)時(shí)器無效耙蔑,并且當(dāng)應(yīng)用停止運(yùn)行時(shí)存儲(chǔ)足夠的應(yīng)用程序狀態(tài)信息以將應(yīng)用程序恢復(fù)到其之前狀態(tài)。
- (void)applicationDidEnterBackground:(UIApplication *)application {
//保存用戶數(shù)據(jù)
BOOL success = [[HQLItemStore sharedStore] saveChanges];
if (success) {
NSLog(@"Saved all of the HQLItem");
}else {
NSLog(@"Could not save any of the HQLItem");
}
}
應(yīng)用打開前讀取數(shù)據(jù)
為了能在Homepwner啟動(dòng)時(shí)載入之前保存的全部Item對(duì)象孤荣,需要在創(chuàng)建Itemstore對(duì)象時(shí)使用NSKeyedUnarchiver類甸陌。
在HQLItemStore.m中,修改初始化方法:
//這是真正的(私有的)初始化方法
- (instancetype)initPrivate {
self = [super init];
//父類的init方法是否成功創(chuàng)建了對(duì)象
if (self) {
//載入之前保存的全部HQLItem對(duì)象
NSString *path = [self itemArchivePath];
//根據(jù)路徑載入固化文件
_privateItems = [NSKeyedUnarchiver unarchiveObjectWithFile:path];
//如果之前沒有保存過_privateItems盐股,就創(chuàng)建一個(gè)新的
if (!_privateItems) {
_privateItems = [[NSMutableArray alloc] init];
}
}
return self;
}