【IOS学习基础】归档和解档

1、归档介绍

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

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

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

  4.归档的缺点:归档的形式来保存数据,只能一次性归档保存以及一次性解压。因此只能针对小量数据,并且对数据操做比较笨拙,即若是想改动数据的某一小部分,仍是须要解压整个数据或者归档整个数据。xcode

2、XML归档

  1.局限:数据类型只支持 NSString、NSDictionary、NSArayy、NSData、NSNumber(若是你想的话,能够将基本数据类型转换为NSNumber再进行归档)。浏览器

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

  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归解档 ****/
// 任何对象均可以转化为NSData
NSData *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];

3、NSKeyedArchiver归档

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

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

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

  Student类


#import <Foundation/Foundation.h>
#import "Book.h"

@interface Student : NSObject<NSCoding> @property (nonatomic,copy)NSString *name; @property (nonatomic,assign)int age; // 除Student类以外,这里有一个自定义类型Book,因此Book类也须要实现NSCoding协议,从而进行归档
@property (nonatomic,strong)Book *book; @end
#import "Student.h"

@implementation Student -(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"]; } return self; } @end

 一样的,Book类也要实现<NSCoding>协议并重写相应的两个方法。

 NSKeyedArchiver归档实现代码:

// 这里student、book对象初始化的代码就不列了
NSArray *stuArr = @[stu1,stu2]; if ([NSKeyedArchiver archiveRootObject:stuArr toFile:path5]) { NSLog(@"写入成功"); } NSArray *arr = [NSKeyedUnarchiver unarchiveObjectWithFile:path5];

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

4、NSUserDefaults

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

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

// 保存
NSString *passWord = @"88888888"; NSUserDefaults *user = [NSUserDefaults standardUserDefaults]; [user setObject:passWord forKey:@"passWord"]; // 经过存储时候key取出相应的value
NSString *passWord = [ user objectForKey:@"passWord"];

  3.固然,NSUserDefaults也能够存储自定义类,同NSKeyedArchiver归档类似,为自定义类实现<NSCoding>协议,而后

// 将student类型变为NSData类型 (这里采用NSKeyedArchiver中的例子Student为例)
NSData *data = [NSKeyedArchiver archivedDataWithRootObject:student];
// NSUserDefaults是支持NSData类型
NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
[user setObject:data forKey:@"student"]; 

5、由sqlite到FMDB

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

  

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

  

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

  若是你想要使用老版本的,请继续下面的操做。

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

  

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

 User类:

#import <Foundation/Foundation.h>

@interface User : NSObject @property (nonatomic,retain)NSData *icon; // 头像
@property (nonatomic,copy)NSString *name; // 名字
@property (nonatomic,copy)NSString *phone;  // 电话号码
@property (nonatomic,assign)int age; // 年龄
 @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 <Foundation/Foundation.h>
#import <sqlite3.h>
#import "User.h"

#define insert @"INSERT"

@interface UserDBManager : 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"

@interface UserDBManager () @property (nonatomic,copy)NSString *dbPath; @property (nonatomic,assign)sqlite3 *dataBase; @end

@implementation UserDBManager -(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]) { //若是存在就打开
        bool result = sqlite3_open([dbPath UTF8String], &_dataBase) == SQLITE_OK; if (result) { return YES; } else { sqlite3_close(_dataBase); return NO; } } 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变量
            int pResult =  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(@"建立表失败"); } } //释放statement
 sqlite3_finalize(stateMent); return YES; } else { sqlite3_close(_dataBase); return NO; } } return YES; } -(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变量
        int result = 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(@"插入失败"); } } //释放statement
 sqlite3_finalize(stateMent); //关闭数据库
 sqlite3_close(_dataBase); return YES; } return NO; } -(BOOL)delePerson:(NSInteger)idd { if ([self openDB]) { NSString *deleSql = [NSString stringWithFormat:@"delete from User where id = %ld",idd];  // 经过id删除user表中的用户数据
        sqlite3_stmt *stateMent; int result = sqlite3_prepare_v2(_dataBase, [deleSql UTF8String], -1, &stateMent, nil); if (result == SQLITE_OK) { //执行sql 语句 成功
            if (sqlite3_step(stateMent) == SQLITE_DONE) { NSLog(@"删除成功"); } else { NSLog(@"删除失败"); } } //释放statement
 sqlite3_finalize(stateMent); //关闭数据库
 sqlite3_close(_dataBase); return YES; } return NO; } -(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变量
        int pResult =  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]; int num = sqlite3_column_int(statement, 0); char *cName = (char *)sqlite3_column_text(statement, 1); char *cPhone = (char *)sqlite3_column_text(statement, 2); int age = 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]; } //释放statement
 sqlite3_finalize(statement); } //关闭数据库
 sqlite3_close(_dataBase); } return arr; } -(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变量
        int pResult =  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]; int num = sqlite3_column_int(statement, 0); char *cName = (char *)sqlite3_column_text(statement, 1); char *cPhone = (char *)sqlite3_column_text(statement, 2); int age = 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); //释放statement
 sqlite3_finalize(statement); } //关闭数据库
 sqlite3_close(_dataBase); } return arr; } @end

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

  4.界面的代码就不贴了,大体是这样的

 

 

 经过前往文件夹,能够看到目录下面有一个后缀名为db的文件

  

打开以后,也能够看到其中的三条数据(这里我使用的是mac山观点SQLite.app软件)

原本准备把FMDB的使用介绍一块儿写在这里的,但想了一下,仍是待之后再写在三方库学习归类里面吧,多复习老是对本身有好处的。(FMDB是通过别人封装好的数据库操做三方库,使用起来很是简单)

 

6、coreData

  还没去了解如何使用,改天再本身学一下再写吧。

相关文章
相关标签/搜索