一、iOS數(shù)據(jù)持久化方式
(1)XML屬性列表(plist)歸檔
(2)Preference(偏好設(shè)置),本質(zhì)還是通過(guò)“plist”來(lái)存儲(chǔ)數(shù)據(jù), 但是使用更簡(jiǎn)單(無(wú)需關(guān)注文件、文件夾路徑和名稱)
(3)NSKeyedArchiver歸檔(NSCoding),可以把任何對(duì)象, 直接保存為文件的方式。
(4)SQLite3,當(dāng)非常大量的數(shù)據(jù)存儲(chǔ)時(shí)使用
(5)Core Data,就是對(duì)SQLite的封裝
關(guān)于bundle路徑和sandbox沙河路徑:
(1)bundle路徑:應(yīng)用程序 (APP) 在手機(jī)里面的安裝路徑
(2)沙河路徑:專門用來(lái)存儲(chǔ)App自己數(shù)據(jù)的一個(gè)路徑,iOS為每個(gè)app都分配了一個(gè)專門用來(lái)存儲(chǔ)這個(gè)app自身的一些數(shù)據(jù)的路徑
二、應(yīng)用沙盒(應(yīng)用程序的文件夾)
1、打印沙盒路徑
NSLog(@"%@",NSHomeDirectory());
2、使用Documents目錄進(jìn)行數(shù)據(jù)持久化的保存,我們平時(shí)操作數(shù)據(jù)主要使用Documents目錄
NSString *path = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"test.plist"];
- 參數(shù)1:第一個(gè)參數(shù)指定了搜索的路徑名稱,NSDocumentDirectory表示是在Documents中尋找,NSCachesDirectory的話就是在cache文件夾中尋找
常用枚舉:
NSDocumentDirectory
NSCachesDirectory
- 參數(shù)2:
NSUserDomainMask = 1,//用戶主目錄中
NSLocalDomainMask = 2,//當(dāng)前機(jī)器中
NSNetworkDomainMask = 4,//網(wǎng)絡(luò)中可見(jiàn)的主機(jī)
NSSystemDomainMask = 8,//系統(tǒng)目錄,不可修改(/System)
NSAllDomainsMask = 0x0ffff,//全部 - 參數(shù)3:是否展開(kāi)波浪線,一般為YES展開(kāi)
Documents:
需要保存由應(yīng)用程序本身產(chǎn)生的文件或者數(shù)據(jù),例如:游戲進(jìn)度、涂鴉軟件的繪圖
目錄中的文件會(huì)被自動(dòng)保存在 iCloud
注意:不要保存從網(wǎng)絡(luò)上下載的文件,否則會(huì)無(wú)法上架!
tmp:
保存臨時(shí)文件,后續(xù)不需要使用
tmp目錄中的文件,系統(tǒng)會(huì)自動(dòng)清理
重新啟動(dòng)手機(jī),tmp 目錄會(huì)被清空
系統(tǒng)磁盤空間不足時(shí),系統(tǒng)也會(huì)自動(dòng)清理
路徑獲取:NSString *tmp = NSTemporaryDirectory();
Library/Caches:
保存臨時(shí)文件,后續(xù)需要使用,例如:緩存圖片,離線數(shù)據(jù)(地圖數(shù)據(jù))
系統(tǒng)不會(huì)清理cache目錄中的文件
就要求程序開(kāi)發(fā)時(shí),必須提供cache目錄的清理解決方案
路徑獲取:利用NSSearchPathForDirectoriesInDomains
函數(shù)(將函數(shù)的第2個(gè)參數(shù)改為:NSCachesDirectory
即可)
Library/Preference:
保存應(yīng)用的所有偏好設(shè)置,使用 NSUserDefault直接讀寫,iOS的Settings(設(shè)置)應(yīng)用會(huì)在該目錄中查找應(yīng)用的設(shè)置信息。iTunes同步設(shè)備時(shí)會(huì)備份該目錄。該目錄由系統(tǒng)管理, 無(wú)需我們來(lái)管理。通常用來(lái)存儲(chǔ)一些基本的軟件配置信息, 比如記住密碼、自動(dòng)登錄等。
路徑獲取: 通過(guò)NSUserDefaults類存取該目錄下的設(shè)置信息
三、使用方法
1、屬性列表
- 屬性列表是一種XML格式的文件,拓展名為plist,如果對(duì)象是NSString、NSDictionary、NSArray、NSData、NSNumber等類型,就可以使用,
- 注意:不能存儲(chǔ)自定義對(duì)象,會(huì)失敗的
- 存方法:writeToFile
- 讀方法:如字典, dictionaryWithContentsOfFile
2、偏好設(shè)置
通過(guò)NSUserDefaults就能直接訪問(wèn)軟件的偏好設(shè)置(Library/Preferences)
UserDefaults設(shè)置數(shù)據(jù)時(shí),不是立即寫入,而是根據(jù)時(shí)間戳定時(shí)地把緩存中的數(shù)據(jù)寫入本地磁盤。所以調(diào)用了set方法之后數(shù)據(jù)有可能還沒(méi)有寫入磁盤應(yīng)用程序就終止了,為解決上述問(wèn)題,通過(guò)調(diào)用synchornize方法強(qiáng)制寫入。寫入步驟:
(1) 獲取偏好設(shè)置對(duì)象
NSUserDefaults *userDefault = [NSUserDefaults standardUserDefaults];
(2)寫入
[userDefault setBool:switcher.isOnforKey:@"key_name"];
(3)同步
[userDefault synchronize];
- 讀取步驟:
(1) 獲取偏好設(shè)置對(duì)象
NSUserDefaults *userDefault = [NSUserDefaults standardUserDefaults];
(2)用一個(gè)變量接收
switcher.on = [userDefault boolForKey:@"key_name"];
3、自定義對(duì)象歸檔 NSKeyedArchiver
注意: 必須遵守NSCoding協(xié)議的對(duì)象才可以進(jìn)行歸檔解檔,默NSString、NSDictionary、NSArray、NSData、NSNumber等類型已遵守NSCoding協(xié)議,可以直接歸檔解檔。
(1)遵守NSCoding協(xié)議,實(shí)現(xiàn)協(xié)議方法
NSCoding協(xié)議中兩個(gè)方法,一般寫在模型中:
- 歸檔調(diào)用
一般在這個(gè)方法里面指定如何歸檔對(duì)象中的每個(gè)實(shí)例變量,可以使用encodeObject:forKey:方法歸檔實(shí)例變量
- (void)encodeWithCoder:(NSCoder *)aCoder;
[encoder encodeObject:self.name forKey:@"name"];
- 解檔調(diào)用
一般在這個(gè)方法里面指定如何解碼文件中的數(shù)據(jù)為對(duì)象的實(shí)例變量,可以使用decodeObject:forKey方法解碼實(shí)例變量
- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder;
self.name = [decoder decodeObjectForKey:@"name"];
initWithCoder原理:只要解析文件就會(huì)調(diào)用,xib,storyboard都是文件,因此只要解析這兩個(gè)文件,就會(huì)調(diào)用initWithCoder,因此如果在storyboard使用自定義view,重寫initWithCoder方法,一定要調(diào)用[super initWithCoder:],因?yàn)橹挥邢到y(tǒng)才知道怎么解析storyboard,如果沒(méi)有調(diào)用,就解析不了這個(gè)文件。
(2)歸檔一個(gè)對(duì)象(先獲取路徑path)
Person *person = [[[Person alloc] init];
[NSKeyedArchiver archiveRootObject:person toFile:path];
(3)解檔一個(gè)對(duì)象
Person *person = [NSKeyedUnarchiver unarchiveObjectWithFile:path];
注意:
(1)如果父類也遵守了NSCoding協(xié)議,應(yīng)該在encodeWithCoder:方法中加上一句[super encodeWithCode:encode];確保繼承的實(shí)例變量也能被編碼,即也能被歸檔
(2)在initWithCoder:方法中加上一句self = [super initWithCoder:decoder];確保繼承的實(shí)例變量也能被解碼,即也能被恢復(fù)
4、多個(gè)對(duì)象歸檔解檔
使用archiveRootObject:toFile:方法可以將一個(gè)對(duì)象直接寫入到一個(gè)文件中,但有時(shí)候可能想將多個(gè)對(duì)象寫入到同一個(gè)文件中,那么就要使用NSData來(lái)進(jìn)行歸檔對(duì)象,NSData可以為一些數(shù)據(jù)提供臨時(shí)存儲(chǔ)空間,以便隨后寫入文件,或者存放從磁盤讀取的文件內(nèi)容。可以使用[NSMutableData data]創(chuàng)建可變數(shù)據(jù)空間
(1) 歸檔步驟
// 新建一塊可變數(shù)據(jù)區(qū)
NSMutableData *data = [NSMutableData data];
// 將數(shù)據(jù)區(qū)連接到一個(gè)NSKeyedArchiver對(duì)象
NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
// 開(kāi)始存檔對(duì)象,存檔的數(shù)據(jù)都會(huì)存儲(chǔ)到NSMutableData中
[archiver encodeObject:person1 forKey:@"person1"];
[archiver encodeObject:person2 forKey:@"person2"];
// 存檔完畢(一定要調(diào)用這個(gè)方法)
[archiver finishEncoding];
// 將存檔的數(shù)據(jù)寫入文件
[data writeToFile:path atomically:YES];
(2) 解檔步驟
// 從文件中讀取數(shù)據(jù)
NSData *data = [NSData dataWithContentsOfFile:path];
// 根據(jù)數(shù)據(jù),解析成一個(gè)NSKeyedUnarchiver對(duì)象
NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
Person *person1 = [unarchiver decodeObjectForKey:@"person1"];
Person *person2 = [unarchiver decodeObjectForKey:@"person2"];
// 恢復(fù)完畢
[unarchiver finishDecoding];