保存和讀取數(shù)據(jù)的機制
在之前寫的Homeowner基礎(chǔ)上更新,通過固化來保存和讀取模型對象,當(dāng)用戶重新運行Homeowner應(yīng)用時,可以讀取之前創(chuàng)建并保存的模型對象。
之前的應(yīng)用:《iOS編程(第四版)》Demo8:Homeowner
固化機制
固化 是由iOS SDK提供的一種保存和讀取對象的機制。當(dāng)應(yīng)用固化某個對象時,會將該對象的所有屬性存入指定的文件夾。當(dāng)應(yīng)用解固某個對象時,會從指定的文件讀取相應(yīng)的數(shù)據(jù),然后根據(jù)數(shù)據(jù)還原對象。
為了固化或解固某個對象,相應(yīng)對象的類必須遵守 NSCoding 協(xié)議,并實現(xiàn)兩個必須方法:
@protocol NSCoding
- (void)encodeWithCoder:(NSCoder *)aCoder;
- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder; // NS_DESIGNATED_INITIALIZER
@end
- 在Item.m中實現(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)用沙盒機制
-
APP的沙盒文檔結(jié)構(gòu)
每個iOS應(yīng)用都有自己專屬的應(yīng)用沙盒(sandbox)。應(yīng)用沙盒就是文件系統(tǒng)中的目錄,但是iOS系統(tǒng)會將每個應(yīng)用的沙盒目錄與文件系統(tǒng)的其他部分隔離。應(yīng)用只能訪問各自的沙盒。
應(yīng)用沙盒目錄
應(yīng)用程序包(application bundle)
包含應(yīng)用可執(zhí)行文件和所有資源文件,例如NIB文件和圖像文件。它是只讀目錄。Doucments/
存放應(yīng)用運行時生成的并且需要保留的數(shù)據(jù)。iTune或iCloud會在同步設(shè)備時備份該目錄。當(dāng)設(shè)備發(fā)生故障時,可以從iTunes或iCloud恢復(fù)該目錄中的文件。例如,Homepwner應(yīng)用可將用戶所擁有的物品信息保存在Documents/中Library/Caches/
存放應(yīng)用運行時生成的需要保留的數(shù)據(jù)。與Documents/目錄不同的是,iTunes或iCloud不會在同步設(shè)備時備份該目錄。不備份緩存數(shù)據(jù)的主要原因是相關(guān)數(shù)據(jù)的體積可能會很大,從而延長同步設(shè)備所需的時間。如果數(shù)據(jù)源是在別處(例如web服務(wù)器),就可以將得到的數(shù)據(jù)保存在Library/Caches/目錄。當(dāng)用戶需要恢復(fù)設(shè)備時,相關(guān)的應(yīng)用只需要從數(shù)據(jù)源(例如web服務(wù)器)再次獲取數(shù)據(jù)即可。Library/Preferences/
存放所有的偏好設(shè)置,iOS的設(shè)置(Setting)應(yīng)用也會在該目錄中查找應(yīng)用的設(shè)置信息。使用NSUserDefaults類,可以通過Library/Preferences/目錄中的某個特定文件以鍵值對形式保存數(shù)據(jù)。iTunes或iCloud會在同步設(shè)備時備份該目錄。tmp/
存放應(yīng)用運行時所需的臨時數(shù)據(jù)。當(dāng)某個應(yīng)用沒有運行時,iOS系統(tǒng)可能會清除該應(yīng)用的 tmp/ 目錄下的文件,但是為了節(jié)約用戶設(shè)備空間,不能依賴這種自動清除機制,而是當(dāng)應(yīng)用不再需要使用 tmp/ 目錄中的文件時,就及時手動刪除這寫文件。iTune或iCloud不會在同步設(shè)備時備份 tmp/ 目錄。通過NSTemporaryDirectory
函數(shù)可以得到應(yīng)用沙盒中的 tmp/ 目錄的全路徑。
獲取文件路徑
在實現(xiàn)保存和讀取模型對象的功能之前,需要先獲取相應(yīng)文件的全路徑(Doucments/)
在HQLItemStore.m中編寫一個實現(xiàn)獲取路徑的方法:
- (NSString *) itemArchivePath {
//NSSearchPathForDirectoriesInDomains:獲取沙盒中某種目錄的全路徑
//注意第一個參數(shù)是NSDocumentDirectory而不是NSDocumentationDirectory
NSArray *documentDirectiorise = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
//從documentDirectiorise數(shù)組獲取第一個,也是唯一文檔目錄路徑
NSString *documentDirectiory = [documentDirectiorise firstObject];
return [documentDirectiory stringByAppendingPathComponent:@"items.archive"];
}
- 啰嗦一下:
這里是用的 HQLItemStore 倉庫類來統(tǒng)一管理 Item,因此這個獲取路徑的方法使用的是 實例方法,嗯,并沒有任何問題。
試想,如果有一個App需要保存用戶賬戶信息,而且一個 App 就只有一條用戶信息,于是我們就把這個管理用戶賬戶的 Person 類設(shè)置為單例類,但是卻沒有必要創(chuàng)建倉庫類了,因為就只有一條用戶信息嘛,于是獲取沙盒路徑的方法也這樣寫?寫在 Person 類里面?也寫成實例方法?可能就有點問題了。
// 單例類方法
+ (instancetype)sharedUser{
static Person *sharedUser = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// 根據(jù)路徑載入固化文件
NSString *path = [self itemArchivePath];
// ToDo
});
return sharedUser;
}
因為當(dāng)從文件中解固之前要獲取對象,對象還沒有創(chuàng)建,先要得到路徑,怎么能用實例化方法呢?
- 解決方法是使用 C 函數(shù):
頭文件聲明:
#import <Foundation/Foundation.h>
// 聲明輔助函數(shù),用于返回文件路徑
NSString *docPath(void);
// ...
實現(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)時,通過NSKeyedArchiver類保存Item模型對象。
在HQLItemStore.h中聲明一個用于保存數(shù)據(jù)的新方法:
- (BOOL) saveChanges;
在HQLItemStore.m中實現(xiàn)該方法:
- (BOOL) saveChanges{
//獲取文件路徑
NSString *path = [self itemArchivePath];
//如果固話成功就返回YES
return [NSKeyedArchiver archiveRootObject:self.privateItems toFile:path];
}
- 在AppDelegate.m頂部導(dǎo)入HQLItemStore.h,然后實現(xiàn)
- (void)applicationDidEnterBackground:
方法,使用此方法可以釋放共享資源,保存用戶數(shù)據(jù),使計時器無效,并且當(dāng)應(yīng)用停止運行時存儲足夠的應(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啟動時載入之前保存的全部Item對象,需要在創(chuàng)建Itemstore對象時使用NSKeyedUnarchiver類。
在HQLItemStore.m中,修改初始化方法:
//這是真正的(私有的)初始化方法
- (instancetype)initPrivate {
self = [super init];
//父類的init方法是否成功創(chuàng)建了對象
if (self) {
//載入之前保存的全部HQLItem對象
NSString *path = [self itemArchivePath];
//根據(jù)路徑載入固化文件
_privateItems = [NSKeyedUnarchiver unarchiveObjectWithFile:path];
//如果之前沒有保存過_privateItems,就創(chuàng)建一個新的
if (!_privateItems) {
_privateItems = [[NSMutableArray alloc] init];
}
}
return self;
}