iOS 編程:Archive保存、讀取與應(yīng)用狀態(tài)、沙盒機制

保存和讀取數(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;
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,362評論 6 537
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,013評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,346評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,421評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 72,146評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,534評論 1 325
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,585評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,767評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,318評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,074評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,258評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,828評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,486評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,916評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,156評論 1 290
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,993評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 48,234評論 2 375

推薦閱讀更多精彩內(nèi)容