一、數據持久化概述
數據持久化就是數據的永久存儲。其本質是將數據保存為文件,存到程序的沙盒中。
1、數據持久化的方式
1.1 writeToFile:簡單對象寫入文件
1.2 NSUserDefaults:應用程序偏好設置
1.3 Sqlite:輕量級關系型數據庫,不能直接存儲對象(NSData除外),需要用到一些SQL語句,先將復雜對象歸檔(對象->NSData)
1.4 CoreData:對象型數據庫,實質是將數據庫的內部存儲細節封裝
1.5 Plist文件
2、應用程序沙盒
每一應用程序都有自己的應用沙盒,沙盒的本質就是一個文件夾,名字是隨機分配的。
與其他應用程序沙盒隔離,應用程序本身只能訪問自己沙盒的數據。(iOS8+對沙盒之間的訪問部分開放)
2.1應用程序包(.app)
包含了應用程序中所用到的所有資源文件和可執行文件(Base on Unix)。iOS8時,app不存儲在沙盒中,有單獨的文件夾存儲所有程序的app包。
2.2 HomeDirectory
Documents:保存應用運行時生成的需要持久化的數據,iTunes同步設備時會備份該目錄。例如,游戲應用可將游戲存檔保存在該目錄
tmp:保存應用運行時所需的臨時數據,使用完畢后再將相應的文件從該目錄刪除。應用沒有運行時,系統也可能會清除該目錄下的文件。iTunes同步設備時不會備份該目錄
Library/Caches:保存應用運行時生成的需要持久化的數據,iTunes同步設備時不會備份該目錄。一般存儲體積大、不需要備份的非重要數據
Library/Preference:保存應用的所有偏好設置,iOS的Settings(設置)應用會在該目錄中查找應用的設置信息。iTunes同步設備時會備份該目錄
2.3 獲取沙盒路徑
沙盒根目錄:NSHomeDirectory();
沙盒臨時目錄:NSTemporaryDirectory();
Library/Preferences:NSUserDefaults
//1.獲取沙盒中Documents文件夾的路徑
//第一種方式:
NSString*documentPath =NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,NSUserDomainMask,YES).lastObject;//NO, path = @"~/"(相對路徑); YES 絕對路徑
NSLog(@"%@", documentPath);
//第二種方式: (不建議采用,因為新版本的操作系統可能會修改目錄名)
NSString*homePath =NSHomeDirectory();
NSString*documentPath2 = [homePathstringByAppendingPathComponent:@"library/caches"];
NSLog(@"%@", documentPath2);
//2.獲取應用程序包路徑(.app)
NSLog(@"%@", [NSBundlemainBundle].resourcePath);
二、簡單對象持久化
1、簡單對象
NSString\NSArray\NSDictionary\NSData
使用writeToFile:方法,將數據存儲為.plist文件
atomically參數為是否寫入緩存
//字符串
NSString*string =@"I?U";
//數組
NSArray*array =@[@"張三",@"李四",@"王五"];
//字典
NSDictionary*dictionary =@{@"name":@"張三",@"age":@"20",@"sex":@"男"};
//NSData
UIImage*image = [UIImageimageNamed:@"1.jpg"];
NSData*data =http://my.oschina.net/zooyf/blog/UIImageJPEGRepresentation(image,1);
//1.拼接存儲路徑
NSString*strPath = [documentPathstringByAppendingPathComponent:@"string.txt"];
NSString*arrayPath = [documentPathstringByAppendingPathComponent:@"array.txt"];
NSString*dicPath ? = [documentPathstringByAppendingPathComponent:@"dict.txt"];
NSString*dataPath? = [documentPathstringByAppendingPathComponent:@"data.txt"];
//2.寫入文件
[stringwriteToFile:strPathatomically:YESencoding:NSUTF8StringEncodingerror:nil];
[arraywriteToFile:arrayPathatomically:YES];
[dictionarywriteToFile:dicPathatomically:YES];
[datawriteToFile:dataPathatomically:YES];
//3.讀取文件內容
NSString*fileString = [NSStringstringWithContentsOfFile:strPathencoding:NSUTF8StringEncodingerror:nil];
NSArray*fileArray = [NSArrayarrayWithContentsOfFile:arrayPath];
NSDictionary*fileDict = [NSDictionarydictionaryWithContentsOfFile:dicPath];
NSData*fileData = http://my.oschina.net/zooyf/blog/[NSDatadataWithContentsOfFile:dataPath];
NSLog(@"%@", fileString);
NSLog(@"%@", fileArray);
NSLog(@"%@", fileDict);
NSLog(@"%@", fileData);
2、文件管理類:NSFileManager
2.1、功能
NSFileManager使用defaultManager創建單例對象。可以創建文件夾,可以刪除、移動、創建文件,判斷文件是否存在。
2.2、使用
//緩存文件夾所在路徑
NSString*cachesPath =NSSearchPathForDirectoriesInDomains(NSCachesDirectory,NSUserDomainMask,YES).lastObject;
NSLog(@"%@", cachesPath);
//在cachesPath路徑下創建一個文件夾
NSString*directoryPath = [cachesPathstringByAppendingPathComponent:@"path"];
NSFileManager*fileManager = [NSFileManagerdefaultManager];//創建文件管理類單例對象
//根據路徑創建文件夾
NSDictionary*fileDate =@{@"createTime":@"2015-9-9"};
[fileManagercreateDirectoryAtPath:directoryPathwithIntermediateDirectories:YESattributes:fileDateerror:nil];
//根據路徑創建文件(只能寫入NSData類型的數據)
[fileManagercreateFileAtPath:directoryPathcontents:dataattributes:fileDate];
//刪除文件
[fileManagerremoveItemAtPath:dicPatherror:nil];//刪除~/documents/dict.txt
3、NSUserDefaults
NSUserDefaults*defaults = [NSUserDefaultsstandardUserDefaults];//單例
[defaultssetValue:@"yfyfyfyfyfyfyfy"forKey:@"username"];
[defaultssetValue:@"123"forKey:@"password"];
//注意:UserDefaults設置數據時,不是立即寫入,而是根據時間戳定時地把緩存中的數據寫入本地磁盤。所以調用了set方法之后數據有可能還沒有寫入磁盤應用程序就終止了。出現以上問題,可以通過調用synchornize方法強制寫入到文件中
[defaultssynchronize];
//讀取
NSString*name = [defaultsvalueForKey:@"username"];
NSString*pwd = [defaultsvalueForKey:@"password"];
二、復雜對象持久化(NSKeyedArchiver)
1、復雜對象
復雜對象是在Foundation框架內不存在的數據類,無法通過writeToFile寫入到文件內,且至少包含一個實例對象。
由于復雜對象無法通過writeToFile:方法寫入文件,只能將復雜對象轉化為NSData對象,再進行數據持久化。
2、NSCoding協議
@protocolNSCoding
- (void)encodeWithCoder:(NSCoder*)aCoder;
- (id)initWithCoder:(NSCoder*)aDecoder;// NS_DESIGNATED_INITIALIZER
@end
3、復雜對象寫入文件
Person.h
//復雜對象歸檔一:遵守NSCoding協議
@interfacePerson :NSObject
@property(nonatomic,assign)NSIntegerage;
@property(nonatomic,retain)NSString*name;
@property(nonatomic,retain)NSString*gender;
@end
Person.m
#import"Person.h"
@implementationPerson//實現NSCoding協議
#pragma mark --進行編碼--
- (void)encodeWithCoder:(NSCoder*)coder
{
//??? [super encodeWithCode:coder];如果父類也遵守了NSCoding協議,確保繼承的實例變量也能被編碼,即也能被歸檔
[coderencodeObject:self.nameforKey:@"name"];
[coderencodeInteger:self.ageforKey:@"age"];
[coderencodeObject:self.genderforKey:@"gender"];
}
#pragma mark --進行解碼--
- (id)initWithCoder:(NSCoder*)aDecoder
{
//??? self = [super initWithCoder:aDecoder];確保繼承的實例變量也能被解碼,即也能被恢復
self= [superinit];
if(self) {
self.name= [aDecoderdecodeObjectForKey:@"name"];
self.gender= [aDecoderdecodeObjectForKey:@"gender"];
self.age= [aDecoderdecodeIntegerForKey:@"age"];
}
returnself;
}
@end
ViewController.m
類方法進行編碼\解碼(只能歸檔一個對象):
NSString*objPath = [cachesPathstringByAppendingPathComponent:@"person.txt"];
[NSKeyedArchiverarchiveRootObject:persontoFile:objPath];
Person*p2 = [NSKeyedUnarchiverunarchiveObjectWithFile:objPath];
實例方法(可以歸檔多個對象):
#pragma mark --對復雜對象進行持久化(歸檔\編碼) --
//過程:(復雜對象->歸檔->NSData->writeToFile:)
Person*person = [[Personalloc]init];
person.name=@"yf";
person.age=20;
person.gender=@"man";
NSMutableData*mtData = http://my.oschina.net/zooyf/blog/[NSMutableDatadata];
//創建歸檔器
NSKeyedArchiver*archiver = [[NSKeyedArchiveralloc]initForWritingWithMutableData:mtData];
//進行歸檔
[archiverencodeObject:personforKey:@"person"];
//***結束歸檔
[archiverfinishEncoding];
//將歸檔之后的mtData寫入文件
NSString*personPath = [cachesPathstringByAppendingPathComponent:@"person.txt"];
[mtDatawriteToFile:personPathatomically:YES];
NSLog(@"%@", personPath);
NSLog(@"%@", mtData);
#pragma mark --從文件中讀取復雜對象(反歸檔\恢復\解碼) --//過程:(讀取文件(NSData)->反歸檔->復雜對象)
//讀取
NSData*readData = http://my.oschina.net/zooyf/blog/[NSDatadataWithContentsOfFile:personPath];
//創建反歸檔工具
NSKeyedUnarchiver*unArchiver = [[NSKeyedUnarchiveralloc]initForReadingWithData:readData];
//使用反歸檔工具對readData進行反歸檔
Person*readPerson = [unArchiverdecodeObjectForKey:@"person"];
4、使用NSKeyedArchive進行深復制
比如對一個Person對象進行深復制
//?臨時存儲person1的數據
NSData?*data?=?[NSKeyedArchiver?archivedDataWithRootObject:person1];
//?解析data,生成一個新的Person對象
Student?*person2?=?[NSKeyedUnarchiver?unarchiveObjectWithData:data];
//?分別打印內存地址
NSLog(@"person1:0x%x",?person1);?//?person1:0x7177a60
NSLog(@"person2:0x%x",?person2);?//?person2:0x7177cf0
在IOS應用中數據持久化是客戶端重要的東西。一般除了本地持久化就是與服務器交互數據從網上獲取了,但是如果沒有網絡的情況下。本地持久化是必須的。一般持久化有4種方法。現在一一介紹,對于網絡的就以后介紹。
第一種:實用屬性列表,第二種:對象歸檔,第三種:使用Iphone的嵌入式數據庫(SQLite3)
給予Iphone應用程序沙盒原理,我們保持的數據都是保存在相對應的應用程序的Document文件夾。既然我們把數據放在每一個應用的Document文件夾中,呢我我們怎么得到相應的路徑呢,其實也不是很難。下面是檢索文檔目錄路徑的代碼:
NSArray*paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,NSUserdomainMask, YES);
NSString*documentDirectory = [paths objectAtIndex:0];
常量NSDocumentDirectory表面我們正在查找的Document目錄路徑,常量NSUserDomainmask表明我們希望將搜索限制于我們應用程序的沙盒中。這樣我們就可以得到該數組的第一值,也僅此一值,因為每一個應用程序只有一個Document文件夾。我們得到了Document的路徑,然后和文件名相連接不就是一個完整的路徑了嗎,這用到了stringByAppendingPathComponent方法:
NSString*filepath = [documentDirectorystringbyAppendingPathComponent:@"filename.xxx"];其中filename.xxx為要命名的文件。
在每一個應用程序中還對應一個temp文件夾,我們怎么獲取這個文件夾的路徑呢,也比較簡單:
NSString*tempPath = NSTemporaryDirectory();
NSString*filePath = [tempPathstringByAppendingPathComponent:@"filename.xxx"];
NSUserDefaults類的使用和NSKeyedArchiver有很多類似之處,但是查看NSUserDefaults的定義可以看出,NSUserDefaults直接繼承自NSObject而NSKeyedArchiver 繼承自NSCoder。這意味著NSKeyedArchiver實際上是個歸檔持久化的類,也就可以使用NSCoder類的[encodeObject: (id)objv forKey:(NSString *)key]方法來對數據進行持久化存儲
除了上面的三種方法來保存持久化 數據以外,我們還可以用寫文件到磁盤的方式來保存持久化數據
嵌入式數據庫持久化數據就是把數據保存在iphone的嵌入式數據庫系統SQLite3中,本質上來說,數據庫持久化操作是基于文件持久化基礎之上的。 要使用嵌入式數據庫SQLite3,首先需要加載其動態庫libsqlite3.dylib,這個文件位于/Xcode3.1.4/Platforms/iPhoneOS.platform/Developer/SDKs/iPhoneOS3.1.sdk/usr/lib目錄下
Core Data是一種 穩定,功能全面的持久化工具,和之前的一些持久化 工具相比,他不需要對實體進行歸檔,也就是序列化,而是在數據 模型編輯器中創建一些實體
在代碼中,你不再使用存取方法和修改方法,而是使用鍵值對編碼來設置屬性或者減縮他們的值
那么這些托管對象的活動區域在哪 ?? 他們位于所謂的持久庫中,默認情況下,Core Data應用程序將持久庫實現為存儲在應用程序文檔目錄的sqlite數據庫。
雖然數據是通過sqlite存儲的,但框架中的類將完成加載和保存數據的相關工作。不許要編寫任何sql語句。
其實在Iphone上存儲數據是在iPhone系統中存儲數據的,這里可以包括應用程序和互聯網的一些數據信息。我們可以解析和讀取這些數據。首先我們先了解下數據持久化的幾種方式:首先是屬性列表。nsuserdefaults位于屬性列表的最頂層,我們可以在里面添加一些UI的小單元
還有就是iPhone的應用程序沙盒技術,以及對象的歸檔,以及第三方的SQLlite,還有就是coredata。
首先讓我們說說這個屬性列表:它是存儲相對較小數據量的簡潔方式,并且耗費非常低的工作量。但是這個屬性列表也是有限制的像剛才我說的存儲數據量小的數據,還有就是一些自定義的對象是不能存儲的,只能是支持序列化的對象,想字典,數據,字符串,對了 還是不能添加UI界面的東西。他們可以以兩種不同的方式存儲一個是XML,這個XML是一種可讀性非常強的格式,另一個是內部專門有的2進制的格式,這種方式速度更快寫,數據量更小些。正如我們所說的NSuerdefaults位于屬性列表的頂層,因此我們可以輕松的讀取并且存儲數據。
下面我們說說有幾種情況來說明有些數據是不能存儲的:比如你有大量的幾百兆的數據這樣是不行的,還有是一堆的圖像,你將不會選擇這個方式,因為你任何時候加載一個屬性列表的時候他們是完全在家在內存當中的,知道一個完整的圖片被創建,因此這種方式是不適用隨機訪問的。假如有兩個線程一個是寫入一個是讀取這是不能同步的,同步會出現一些問題的,這是為什么呢,因為屬性列表沒有類似有原子性的東西。我們會有兩種方式把他們寫在硬盤內writetofile:@"文件路徑"atomically:yes。這個原子性是什么意思呢,在這里可以這樣的理解,它可以讓系統寫入文件一次首先是把它寫入到一個臨時的文件后然后在將其轉移到可你以放置任何文件的沒有風險的地方這里將保存著先前的文件以及任何改動的地方,因此如果操作 系統崩潰或者電池電量的耗盡的時候你也不會丟失任何的數據的。你可以從你先前的或者新的數據里獲取數據而非受破壞的中間的狀態。當你在寫一個屬性列表的時候如果你看XML和2進制。自定義和一些UI的視圖不支持這些東西取決與可不可以序列化對象,當然不可以序列化,即使你序列化這些不能序列化的東西它會返回空值的。