一、已有類型的歸檔和解檔 首先來看一個簡單的例子: //第二種方式 //第一種方式的缺陷是一個對象歸檔成一個文件 //但是第二種方式,多個對象可以歸檔成一個文件 NSArray *array = [NSArray arrayWithObjects:@"zhangsan",@"lisi", nil];
一、已有類型的歸檔和解檔
首先來看一個簡單的例子:
//第一方式:歸檔對象 //對象-->文件 NSArray *array = [NSArray arrayWithObjects:@"zhang",@"wangwu",@"lisi",nil]; // NSHomeDirectory 獲取根目錄 stringByAppendingPathComponent 添加儲存的文件名 NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"]; BOOL success = [NSKeyedArchiver archiveRootObject:array toFile:filePath]; if(success){ NSLog(@"保存成功"); }else { NSLog(@"未保存"); } // 解歸檔 array = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath]; NSLog(@"%@",array);
//第二種方式
//第一種方式的缺陷是一個對象歸檔成一個文件
//但是第二種方式,多個對象可以歸檔成一個文件
NSArray *array = [NSArray arrayWithObjects:@"zhangsan",@"lisi", nil];
NSMutableData *data = [NSMutableData data];
NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
//編碼
[archiver encodeObject:array forKey:@"array"];
[archiver encodeInt:100 forKey:@"scope"];
[archiver encodeObject:@"jack" forKey:@"name"];
//完成編碼,將上面的歸檔數據填充到data中,此時data中已經存儲了歸檔對象的數據
[archiver finishEncoding];
NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
BOOL success = [data writeToFile:filePath atomically:YES];
if(success){
NSLog(@"歸檔成功");
}
// 對多個對象進行解檔操作
/*
NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
//讀取歸檔數據
NSData *data = [[NSData alloc] initWithContentsOfFile:filePath];
//創建解歸檔對象,對data中的數據進行解歸檔
NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
//解歸檔
NSArray *array = [unarchiver decodeObjectForKey:@"array"];
NSLog(@"%@",array);
int value = (int)[unarchiver decodeObjectForKey:@"scope"];
NSLog(@"%d",value);
*/
- 歸檔 下麵這段代碼是將一個NSArray對象寫入到一個文件中。
//對象-->文件 NSArray *array = [NSArray arrayWithObjects:@"zhang",@"wangwu",@"lisi",nil]; // NSHomeDirectory 獲取根目錄 stringByAppendingPathComponent 添加儲存的文件名 NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"]; BOOL success = [NSKeyedArchiver archiveRootObject:array toFile:filePath]; if(success){ NSLog(@"保存成功"); }else { NSLog(@"未保存");
下麵代碼是創建一個文件的方法
NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
- 解檔 下麵代碼是解檔就是返回一個對象
array = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath]; NSLog(@"%@",array);
- 對多個對象進行歸檔到一個文件
//第二種方式 //第一種方式的缺陷是一個對象歸檔成一個文件 //但是第二種方式,多個對象可以歸檔成一個文件 NSArray *array = [NSArray arrayWithObjects:@"zhangsan",@"lisi", nil]; NSMutableData *data = [NSMutableData data]; NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data]; //編碼 [archiver encodeObject:array forKey:@"array"]; [archiver encodeInt:100 forKey:@"scope"]; [archiver encodeObject:@"jack" forKey:@"name"]; //完成編碼,將上面的歸檔數據填充到data中,此時data中已經存儲了歸檔對象的數據 [archiver finishEncoding]; NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"]; BOOL success = [data writeToFile:filePath atomically:YES]; if(success){ NSLog(@"歸檔成功"); }
多個對象歸檔的話,這裡要用到一個類:NSMutableData和NSData,他們兩的區別很簡單,一個是可變的,一個是不可變的。然後這裡還創建了一個歸檔器:NSKeyedArchiver,這個類負責進行指定類型的編碼操作,然後將數據填充到NSMutableData類。歸檔的時候對每個類型對象用一個key進行對應,這個NSData和NSDirctionary很類似了。
- 對多個對象進行解檔操作
NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"]; //讀取歸檔數據 NSData *data = [[NSData alloc] initWithContentsOfFile:filePath]; //創建解歸檔對象,對data中的數據進行解歸檔 NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data]; //解歸檔 NSArray *array = [unarchiver decodeObjectForKey:@"array"]; NSLog(@"%@",array); int value = (int)[unarchiver decodeObjectForKey:@"scope"]; NSLog(@"%d",value);
我們可以將文件解檔出一個NSData對象,然後可以通過key去獲取指定的類型對象
二、自定義類型的歸檔和解檔
上面說到了已有類型的歸檔和解檔,下麵來看一下自定義類型的歸檔和解檔操作,在開始的時候也說了,如果自定義的類型可以進行歸檔和解檔的話,必須實現一個協議:NSCoding
Student.h
#import <Foundation/Foundation.h> // 類只有實現NSCoding協議才能歸檔 @interface Student : NSObject<NSCoding> @property(copy,nonatomic)NSString *name; @property(assign,nonatomic)int age; @property(strong,nonatomic) NSString *adder; @end
這裡自定義了一個Student類型,實現了NSCoding協議,然後他有三個屬性,這裡我們看到有新的方法去定義屬性
Student.m
#import "Student.h" @implementation Student // 歸檔時調用 也是一個初始化 - (instancetype)initWithCoder:(NSCoder *)aDecoder { NSLog(@"initWithCoder"); self = [super init]; if (self!=nil) { //一般我們將key定義成巨集,這樣就不會出錯 _name = [[aDecoder decodeObjectForKey:@"name"] copy]; self.age = (int)[aDecoder decodeIntegerForKey:@"age"]; _adder=[aDecoder decodeObjectForKey:@"adder"]; } return self; } // 歸檔時調用此方法 - (void)encodeWithCoder:(NSCoder *)aCoder{ NSLog(@"encodeWithCoder"); [aCoder encodeObject:_name forKey:@"name"];//一般key和屬性名是取一樣的 [aCoder encodeInteger:_age forKey:@"age"]; [aCoder encodeObject:_adder forKey:@"adder"]; } // 描述方法 - (NSString *)description { return [NSString stringWithFormat:@"name=%@,age=%d,adder=%@", _name,_age,_adder]; } @end
在Person.m文件中,我們需要實現協議中的兩個方法:
initWithCoder
encodeWithCoder
這兩個方法一個是用於歸檔操作時會調用的方法,還有一個是用於解檔操作時會調用的方法
1、解檔的時候用到的方法
- (instancetype)initWithCoder:(NSCoder *)aDecoder { NSLog(@"initWithCoder"); self = [super init]; if (self!=nil) { //一般我們將key定義成巨集,這樣就不會出錯 _name = [[aDecoder decodeObjectForKey:@"name"] copy]; self.age = (int)[aDecoder decodeIntegerForKey:@"age"]; _adder=[aDecoder decodeObjectForKey:@"adder"]; } return self; }
這個是一個初始化的方法,同時他也是一個解檔操作時會調用的方法,所以在這裡我們既要寫一下初始化方法的特定代碼,還要寫上解檔的代碼,這裡主要看解檔的代碼
其實很簡單,就是對屬性重新寫一下值,然後對每個屬性指定一個key就可以了。
2、歸檔的時候用到的方法
// 歸檔時調用此方法 - (void)encodeWithCoder:(NSCoder *)aCoder{ NSLog(@"encodeWithCoder"); [aCoder encodeObject:_name forKey:@"name"];//一般key和屬性名是取一樣的 [aCoder encodeInteger:_age forKey:@"age"]; [aCoder encodeObject:_adder forKey:@"adder"]; }
歸檔和解檔的操作正好相反的,但是要註意的是:他們屬性的key一定要保持一致
3、重寫description方法
- (NSString *)description { return [NSString stringWithFormat:@"name=%@,age=%d,adder=%@", _name,_age,_adder]; }
在之前的文章中我說道過,我們在使用NSLog方法列印對象的值的時候,其實是調用對象的description方法,而這個方法是NSObject類中的,我們可以重寫他,這樣我們就可以列印我們想要的信息了。和Java中的toString方法一樣。
下麵就來看一下使用方法了
#import "ViewController.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; Student *stu=[Student new]; stu.name=@"張三"; stu.age=12; stu.adder=@"北京"; // 歸檔 NSString *filePath=[NSHomeDirectory() stringByAppendingPathComponent:@"message.plist"]; NSLog(@"%@",filePath); BOOL bol=[NSKeyedArchiver archiveRootObject:stu toFile:filePath]; if (bol) { NSLog(@"歸檔成功"); }else{ NSLog(@"歸檔成功"); } // 解歸檔 Student *stu1=[NSKeyedUnarchiver unarchiveObjectWithFile:filePath]; NSLog(@"%@",stu1); } - (void)didReceiveMemoryWarning { [super didReceiveMemoryWarning]; // Dispose of any resources that can be recreated. } @end
我們可以看到,使用起來是很簡單的和上面的方式一樣,運行結果:
看到了,我們自定義的description方法,列印了我們自己想要的結果~~