归档和解档

一、归档介绍

1.归档是指用某种格式来保存一个或多个对象,以便以后还原这些对象的过程。归档是将数据持久化的一种方式(所谓数据持久化,就是指在IOS开发过程中,将数据保存到本地,能够让程序的运行更加流畅)。

2.想要归档的数据对象,需要遵守NSCoding协议,并且该对象对应的类必须提供encodeWithCoder:initWithCoder:方法。

3.归档就是将临时数据保存成本地文件。

4.归档的缺点:归档的形式来保存数据,只能一次性归档保存以及一次性解压。所以只能针对小量数据,而且对数据操作比较笨拙,即如果想改动数据的某一小部分,还是需要解压整个数据或者归档整个数据。

二、XML归档

1.局限:数据类型只支持 NSString、NSDictionary、NSArayy、NSData、NSNumber(如果你想的话,可以将基本数据类型转换为NSNumber再进行归档)。

2.比较方便,设置好归档路径,一句话归档,一句话解档。

3.归档文件格式:一般保存.plist文件。

/**** NSString和NSMutableString XML归解档 ****/NSString*str =@"hello world";

NSString*path = [[NSHomeDirectory() stringByAppendingPathComponent:@"Desktop"]stringByAppendingPathComponent:@"hello.txt"];//atomically:这个参数意思是如果为YES,则保证文件的写入原子性。就是说会先创建一个临时文件,直到文件内容写入成功再导入到目标文件里.如果为NO,则直接写入目标文件里.[str writeToFile:path atomically:YES encoding:NSUTF8StringEncoding error:nil];//这里会覆盖原来的内容[@"hello world 2"writeToFile:path atomically:YES encoding:NSUTF8StringEncoding error:nil];

NSString*str2 = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:nil];

/**** NSData和NSMutableData XML归解档 ****///任何对象都可以转化为NSDataNSData *data = [@"hello world"dataUsingEncoding:NSUTF8StringEncoding];

NSString*path2 = [[NSHomeDirectory() stringByAppendingPathComponent:@"Desktop"]stringByAppendingPathComponent:@"data.txt"];//归档[data writeToFile:path2 atomically:YES];//解档[NSData dataWithContentsOfFile:path2];

/**** NSArray及NSMutableArray XML归解档 ****/

NSArray*array = @[@"test",@"test2"];

NSString *path3 = [[NSHomeDirectory() stringByAppendingPathComponent:@"Desktop"]stringByAppendingPathComponent:@"array.plist"];//归档[array writeToFile:path3 atomically:YES];// 解档NSArray *array = [NSArray arrayWithContentsOfFile:path3];

/**** NSArray XML归解档 ****/NSDictionary*dic = [NSDictionary dictionaryWithObjectsAndKeys:@"one",@"1",@"two",@"2", nil];

NSString*path4 = [[NSHomeDirectory() stringByAppendingPathComponent:@"Desktop"]stringByAppendingPathComponent:@"dic.plist"];//归档[dic writeToFile:path4 atomically:YES];//解档NSDictionary *ddic = [NSDictionary dictionaryWithContentsOfFile:path3];

三、NSKeyedArchiver归档

1.将各种类型的对象存储到文件中,而且不仅仅是字符串、数组和字典类型,有一种更灵括的方法。就是利用NSKeyedArchiver类创建带键(keyed)的档案来完成。实现对我们自定义的类进行归档。

2.序列化与反序列化:将一个Objective-C对象转换成NSData的操作叫做对象的序列化;而将一个NSData转换成Objective-C对象的操作叫做对象的反序列化。一个Objective-C对象需要通过实现NSCoding协议以便支持序列化与反序列化

3.模拟场景:有一个学生类,学生拥有三个属性name、age、book(一本书),其中book对应Book类,Book类中拥有一个属性bookName,归档一个数组,数组中有两个student对象。

Student类

#import

#import"Book.h"@interfaceStudent : NSObject@property (nonatomic,copy)NSString*name;

@property (nonatomic,assign)intage;//除Student类之外,这里有一个自定义类型Book,所以Book类也需要实现NSCoding协议,从而进行归档@property (nonatomic,strong)Book *book;@end#import"Student.h"@implementationStudent-(void)encodeWithCoder:(NSCoder *)aCoder

{//归档姓名(NSString 对象)[aCoder encodeObject:self.name forKey:@"name"];//归档年龄(基本数据类型,如果是其它基本数据类型调用相应的encode方法)[aCoder encodeInt:self.age forKey:@"age"];//归档自定义类(Book)[aCoder encodeObject:self.book forKey:@"book"];

}-(instancetype)initWithCoder:(NSCoder *)aDecoder

{if(self =[super init])

{//归档的key 写的什么 对应属性解档key就写什么self.name = [aDecoder decodeObjectForKey:@"name"];

self.age= [aDecoder decodeInt32ForKey:@"age"];

self.book= [aDecoder decodeObjectForKey:@"book"];

}returnself;

}@end

同样的,Book类也要实现

协议并重写相应的两个方法。

NSKeyedArchiver归档实现代码:

//这里student、book对象初始化的代码就不列了NSArray *stuArr =@[stu1,stu2];if([NSKeyedArchiver archiveRootObject:stuArr toFile:path5])

{

NSLog(@"写入成功");

}

NSArray*arr = [NSKeyedUnarchiver unarchiveObjectWithFile:path5];

如此,便可以实现自定义对象的归档和解档了。

四、NSUserDefaults

1.NSUserDefaults是一个单例类,如它的名字一样,用于永久保存一些用户对于应用程序的配置之类的简单数据,其简单而又实用

2.NSUserDefaults支持的数据类型同XML归档一样,仅仅用于保存一些程序配置信息的话完全是可以胜任的。

//保存NSString *passWord =@"88888888";

NSUserDefaults*user =[NSUserDefaults standardUserDefaults];

[user setObject:passWord forKey:@"passWord"];//通过存储时候key取出相应的valueNSString *passWord = [ user objectForKey:@"passWord"];

3.当然,NSUserDefaults也可以存储自定义类,同NSKeyedArchiver归档相似,为自定义类实现

协议,然后

//将student类型变为NSData类型 (这里采用NSKeyedArchiver中的例子Student为例)NSData *data = [NSKeyedArchiver archivedDataWithRootObject:student];

// NSUserDefaults是支持NSData类型

NSUserDefaults *user = [NSUserDefaults standardUserDefaults];

[user setObject:data forKey:@"student"];

五、由sqlite到FMDB

1.使用sqlite之前需要了解一下基本的一些sql语句,很简单,学会建表、增删改查的语句就行了。(学习的话这里推荐火狐浏览器下的sqlite组件)

2.使用:在工程中导入包“libsqlite3.dylib”,在类中导入头文件#import

Xcode7注意了,当你去导入sqlite3.dylib的时候(其实那两个.tbd文件就是以前老版本的替代了),你会发现根本找不到这个包,因为这个包自xcode7之后就被苹果隐藏掉了

如果你想要使用老版本的,请继续下面的操作。

当点击“+”弹出窗口之后,选择“add other”,快捷键 CMD+Shift+G (Go to the folder),输入/usr/lib后,进入隐藏的界面,在文件目录下找到,然后添加你需要的 *.dylib,如libsqlite3.dylib文件。

3.sqlite使用,这里讲一下我之前做的一个增删改查的demo,下面贴上代码

User类:

#import

@interfaceUser : NSObject

@property (nonatomic,retain)NSData*icon;//头像@property (nonatomic,copy)NSString *name;//名字@property (nonatomic,copy)NSString *phone;//电话号码@property (nonatomic,assign)intage;//年龄@property (nonatomic,assign)NSInteger ID;@end

UserDBManager类

////UserDBManager.h//Sqlite3基础操作////Created by silence on 15-8-15.//Copyright (c) 2015年 hawode06. All rights reserved.//#import

#import#import"User.h"#defineinsert @"INSERT"@interfaceUserDBManager : NSObject//插入数据-(BOOL)insertUser:(User *)user;//删除数据-(BOOL)delePerson:(NSInteger)idd;//修改数据-(BOOL)updatePerson:(User *)User;//查询所有-(NSArray *)queryUser;//通过姓名查询用户-(NSArray *)searchWithName:(NSString *)str;@end////UserDBManager.m//Sqlite3基础操作////Created by silence on 15-8-15.//Copyright (c) 2015年 hawode06. All rights reserved.//#import"UserDBManager.h"@interfaceUserDBManager ()

@property (nonatomic,copy)NSString*dbPath;

@property (nonatomic,assign)sqlite3*dataBase;@end@implementationUserDBManager-(BOOL)openDB

{

NSString*path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)[0];

NSString*dbPath = [path stringByAppendingPathComponent:@"user.db"];

NSLog(@"%@",dbPath);

NSFileManager*fm =[NSFileManager defaultManager];if([fm fileExistsAtPath:dbPath])

{//如果存在就打开boolresult = sqlite3_open([dbPath UTF8String], &_dataBase) ==SQLITE_OK;if(result)

{returnYES;

}else{

sqlite3_close(_dataBase);returnNO;

}

}else{//打开数据库 返回值sqlite_ok 代表打开成功if(sqlite3_open([dbPath UTF8String], &_dataBase) ==SQLITE_OK)

{

NSString*sql =@"create table  if not exists \"User\"(\"id\" INTEGER PRIMARY KEY  AUTOINCREMENT  NOT NULL  UNIQUE,\"name\" varchar,\"phone\" varchar,\"age\" INTEGER NOT NULL DEFAULT 1,\"image\" blob)";//执行创建表的sql语句sqlite3_stmt *stateMent;//解析sql语句  第一个 数据库变量  第二个sql语句  sql语句的长度 -1代表自动计算  stetement变量intpResult =  sqlite3_prepare_v2(_dataBase, [sql UTF8String], -1, &stateMent, nil);//解析sql 语句失败if(pResult !=SQLITE_OK)

{

NSLog(@"创建表失败");

}else{//执行sql 语句 成功if(sqlite3_step(stateMent) ==SQLITE_DONE)

{

NSLog(@"创建表成功");

}else{

NSLog(@"创建表失败");

}

}//释放statementsqlite3_finalize(stateMent);returnYES;

}else{

sqlite3_close(_dataBase);returnNO;

}

}returnYES;

}-(BOOL)insertUser:(User *)user

{if([self openDB])

{

NSString*addSql =@"insert into User (name,phone,age,image) values (?,?,?,?)";

sqlite3_stmt*stateMent;//解析sql语句  第一个 数据库变量  第二个sql语句  sql语句的长度 -1代表自动计算  stetement变量intresult = sqlite3_prepare_v2(_dataBase, [addSql UTF8String], -1, &stateMent, nil);//给sql语句里的’?‘  赋值  第一个参数是statement 第二个是‘?’的位置 以1开头//姓名sqlite3_bind_text(stateMent,1, [user.name UTF8String], -1, SQLITE_TRANSIENT);//电话sqlite3_bind_text(stateMent,2, [user.phone UTF8String], -1, SQLITE_TRANSIENT);//年龄sqlite3_bind_int(stateMent,3, user.age);//头像sqlite3_bind_blob(stateMent,4, [user.icon bytes], (int)user.icon.length, nil);if(result ==SQLITE_OK)

{//执行sql 语句 成功if(sqlite3_step(stateMent) ==SQLITE_DONE)

{

NSLog(@"插入成功");

}else{

NSLog(@"插入失败");

}

}//释放statementsqlite3_finalize(stateMent);//关闭数据库sqlite3_close(_dataBase);returnYES;

}returnNO;

}-(BOOL)delePerson:(NSInteger)idd

{if([self openDB])

{

NSString*deleSql = [NSString stringWithFormat:@"delete from User where id = %ld",idd];//通过id删除user表中的用户数据sqlite3_stmt *stateMent;intresult = sqlite3_prepare_v2(_dataBase, [deleSql UTF8String], -1, &stateMent, nil);if(result ==SQLITE_OK)

{//执行sql 语句 成功if(sqlite3_step(stateMent) ==SQLITE_DONE)

{

NSLog(@"删除成功");

}else{

NSLog(@"删除失败");

}

}//释放statementsqlite3_finalize(stateMent);//关闭数据库sqlite3_close(_dataBase);returnYES;

}returnNO;

}-(NSArray *)queryUser

{

NSMutableArray*arr;if([self openDB])

{

arr= [NSMutableArray arrayWithCapacity:0];

NSString*sql =@"select * from User";//sql 语句:查询User表中的所有数据sqlite3_stmt *statement;//解析sql语句  第一个 数据库变量  第二个sql语句  sql语句的长度 -1代表自动计算  stetement变量intpResult =  sqlite3_prepare_v2(_dataBase, [sql UTF8String], -1, &statement, nil);//解析sql语句成功if(pResult ==SQLITE_OK)

{while(sqlite3_step(statement) ==SQLITE_ROW)

{

User*user =[[User alloc] init];intnum = sqlite3_column_int(statement,0);char*cName = (char*)sqlite3_column_text(statement,1);char*cPhone = (char*)sqlite3_column_text(statement,2);intage = sqlite3_column_int(statement,3);char*iconBytes = (char*)sqlite3_column_blob(statement,4);

NSInteger iconDataLen= sqlite3_column_bytes(statement,4);

user.name=[NSString stringWithCString:cName encoding:NSUTF8StringEncoding];

user.phone=[NSString stringWithCString:cPhone encoding:NSUTF8StringEncoding];

user.age=age;

user.icon=[NSData dataWithBytes:iconBytes length:iconDataLen];

user.ID=num;

[arr addObject:user];

}//释放statementsqlite3_finalize(statement);

}//关闭数据库sqlite3_close(_dataBase);

}returnarr;

}-(NSArray *)searchWithName:(NSString *)str

{

NSMutableArray*arr;if([self openDB])

{

arr= [NSMutableArray arrayWithCapacity:0];

NSString*sql = [NSString stringWithFormat:@"select * from User where name like '%%%@%%'",str];

NSLog(@"sql语句:%@",sql);

sqlite3_stmt*statement;//解析sql语句  第一个 数据库变量  第二个sql语句  sql语句的长度 -1代表自动计算  stetement变量intpResult =  sqlite3_prepare_v2(_dataBase, [sql UTF8String], -1, &statement, nil);//解析sql语句成功if(pResult ==SQLITE_OK)

{while(sqlite3_step(statement) ==SQLITE_ROW)

{

User*user =[[User alloc] init];intnum = sqlite3_column_int(statement,0);char*cName = (char*)sqlite3_column_text(statement,1);char*cPhone = (char*)sqlite3_column_text(statement,2);intage = sqlite3_column_int(statement,3);char*iconBytes = (char*)sqlite3_column_blob(statement,4);

NSInteger iconDataLen= sqlite3_column_bytes(statement,4);

user.name=[NSString stringWithCString:cName encoding:NSUTF8StringEncoding];

user.phone=[NSString stringWithCString:cPhone encoding:NSUTF8StringEncoding];

user.age=age;

user.icon=[NSData dataWithBytes:iconBytes length:iconDataLen];

user.ID=num;

[arr addObject:user];

}

NSLog(@"数组%@",arr);//释放statementsqlite3_finalize(statement);

}//关闭数据库sqlite3_close(_dataBase);

}returnarr;

}@end

// 修改用户数据的好像当时懒得做,并没有实现那个方法,看懂了另外几个增、查、删的实现也就差不多了,基本操作数据库的流程都一样,只是执行不同的sql语句而已。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,783评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,360评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 142,942评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,507评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,324评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,299评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,685评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,358评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,652评论 1 293
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,704评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,465评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,318评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,711评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,991评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,265评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,661评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,864评论 2 335

推荐阅读更多精彩内容