• iOS数据持久化存储之归档NSKeyedArchiver


    归档是一种很常用的文件储存方法,几乎任何类型的对象都能够被归档储存(实际上是一种文件保存的形式),收集了网上的一些资料并结合自己的一些经验,总结如下。

    一、使用archiveRootObject进行简单的归档

    使用NSKeyedArichiver进行归档、NSKeyedUnarchiver进行接档,这种方式会在写入、读出数据之前对数据进行序列化、反序列化操作。

    归档:

        //1.获取文件路径  

      NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0];   

      //2、添加储存的文件名

      NSString *path  = [docPath stringByAppendingPathComponent:@"data.archiver"];    

      //3、将一个对象保存到文件中

      BOOL flag = [NSKeyedArchiver archiveRootObject:@”归档” toFile:path];    

    这种方式可以对字符串、数字等进行归档,当然也可以对NSArray与NSDictionary进行归档。返回值Flag标志着是否归档成功,YES为成功,NO为失败。

    接档:

      

      //1.获取文件路径

      NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0];

      NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"]; NSLog(@"path=%@",path);

      //2.从文件中读取对象

      [NSKeyedUnarchiver unarchiveObjectWithFile:path]  

    使用NSKeyedUnarchiver进行接档(反序列化)。

    这种归档的方式存在一个缺点:只能把一个对象归档进一个文件中,那么怎么对多个对象进行归档呢?

    二、对多个对象的归档

    同样是使用NSKeyedArchiver进行归档,不同的是同时归档多个对象,这里我们举例放入了一个CGPoint点、字符串、整数(当然很多类型都可以的,例如UIImage、float等等),使用encodeXXX方法进行归档,最后通过writeToFile方法写入文件。

    归档:写入数据

      //准备数据  
    
      CGPoint point = CGPointMake(1.0, 2.0);  
    
      NSString *info = @"坐标原点";  
    
      NSInteger value = 10;  
    
      NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0];
    
      NSString *multiHomePath = [docPath stringByAppendingPathComponent:@"multi.archiver"];  
    
      NSMutableData *data = [[NSMutableData alloc] init];  
    
      NSKeyedArchiver *archvier = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];  
    
      //对多个对象进行归档  
    
      [archvier encodeCGPoint:point forKey:@"kPoint"];  
    
      [archvier encodeObject:info forKey:@"kInfo"];  
    
      [archvier encodeInteger:value forKey:@"kValue"];  
    
      [archvier finishEncoding];  
    
      [data writeToFile:multiHomePath atomically:YES];  

    接档:从路径中获得数据构造NSKeyedUnarchiver实例,使用decodeXXXForKey方法获得文件中的对象。

      NSMutableData *dataR = [[NSMutableData alloc] initWithContentsOfFile:multiHomePath];  
    
      NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc]initForReadingWithData:dateR];  
    
      CGPoint pointR = [unarchiver decodeCGPointForKey:@"kPoint"];  
    
      NSString *infoR = [unarchiver decodeObjectForKey:@"kInfo"];  
    
      NSInteger valueR = [unarchiver decodeIntegerForKey:@"kValue"];  
    
      [unarchiver finishDecoding];  
    
      NSLog(@"%f,%f,%@,%d",pointR.x,pointR.y,infoR,valueR);  

     

    可以看出对多个对象进行归档还是挺方便的,这里又出现一个问题,这里的对象都是基本类型数据,那么怎么对自己定义类生成的实例对象进行归档呢?

      

    三、对自定义对象进行归档

    自定义对象,应用范围很广,因为它对应着MVC中的Model层,即实体类。在程序中,我们会在Model层定义很多的entity,例如User,Teacher。。

    那么对自定义对象的归档显得重要的多,因为很多情况下我们需要在Home键之后保存数据,在程序恢复时重新加载,那么,归档便是一个好的选择。

    首先我们需要,自定义一个实体类。

    //  YYViewController.m
    
    
    #import "YYViewController.h"
    #import "YYPerson.h"
    
    @interface YYViewController ()
    - (IBAction)saveBtnOnclick:(id)sender;
    - (IBAction)readBtnOnclick:(id)sender;
    
    @end
    
    @implementation YYViewController
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
    }
    
    
    - (IBAction)saveBtnOnclick:(id)sender {
        //1.创建对象
        YYPerson *p=[[YYPerson alloc]init];
        p.name=@"圆圆";
        p.age=23;
        p.height=1.7;
        
        //2.获取文件路径
        NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)lastObject];
        NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"];
        NSLog(@"path=%@",path);
        
        //3.将自定义的对象保存到文件中
        [NSKeyedArchiver archiveRootObject:p toFile:path];
        
    }
    
    - (IBAction)readBtnOnclick:(id)sender {
        //1.获取文件路径
        NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)lastObject];
        NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"];
        NSLog(@"path=%@",path);
        
        //2.从文件中读取对象
        YYPerson *p=[NSKeyedUnarchiver unarchiveObjectWithFile:path];
        NSLog(@"%@,%d,%.1f",p.name,p.age,p.height);
    }
    @end
    //  YYPerson.h
    
    #import <Foundation/Foundation.h>
    
    // 如果想将一个自定义对象保存到文件中必须实现NSCoding协议
    @interface YYPerson : NSObject<NSCoding>
    
    //姓名
    @property(nonatomic,copy)NSString *name;
    //年龄
    @property(nonatomic,assign)int age;
    //身高
    @property(nonatomic,assign)double height;
    @end
    //  YYPerson.m
    
    #import "YYPerson.h"
    
    @implementation YYPerson
    
    // 当将一个自定义对象保存到文件的时候就会调用该方法
    // 在该方法中说明如何存储自定义对象的属性
    // 也就说在该方法中说清楚存储自定义对象的哪些属性
    -(void)encodeWithCoder:(NSCoder *)aCoder
    {
        NSLog(@"调用了encodeWithCoder:方法");
        [aCoder encodeObject:self.name forKey:@"name"];
        [aCoder encodeInteger:self.age forKey:@"age"];
        [aCoder encodeDouble:self.height forKey:@"height"];
    }
    
    // 当从文件中读取一个对象的时候就会调用该方法
    // 在该方法中说明如何读取保存在文件中的对象
    // 也就是说在该方法中说清楚怎么读取文件中的对象
    -(id)initWithCoder:(NSCoder *)aDecoder
    {
        NSLog(@"调用了initWithCoder:方法");
        //注意:在构造方法中需要先初始化父类的方法
        if (self=[super init]) {
            self.name=[aDecoder decodeObjectForKey:@"name"];
            self.age=[aDecoder decodeIntegerForKey:@"age"];
            self.height=[aDecoder decodeDoubleForKey:@"height"];
        }
        return self;
    }
    @end

    基于归档创建一个用于本地数据存储的类如下:

    #import <Foundation/Foundation.h>
    
    @interface LocalArchiverManager : NSObject
    
    /**单例模式,获取请求管理类
     *param        param:   无
     *
    eturns      return:     无
     */
    + (LocalArchiverManager *)shareManagement;
    
    /**清除本地的序列化的文件
     *param        param:   无
     *
    eturns      return:     无
     */
    - (void)clearArchiverData;
    
    /**保存缓存数据
     *param        obj:   数据源
     *param        key:     接口的名称
     *
    eturns      无
     */
    - (void)saveDataArchiver:(id)obj andAPIKey:(NSString *)key;
    
    /**回去缓存数据
     *param        obj:   api的key
     *
    eturns      id:     返回的数据源
     */
    - (id)archiverQueryAPIKey:(NSString *)key;
    @end
    LocalArchiverManager.m 文件
    #import "LocalArchiverManager.h"
    
    static LocalArchiverManager *m_localArchiverMana;
    #define Document [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0]
    #define ArchiverFile    [Document stringByAppendingPathComponent:@"Archiver"]
    
    @interface LocalArchiverManager()
    
    @property(nonatomic,retain)NSFileManager *fileManager;
    
    @end
    
    @implementation LocalArchiverManager+ (LocalArchiverManager *)shareManagement
    {
        static dispatch_once_t onceTocken;
        dispatch_once(&onceTocken, ^
        {
            m_localArchiverMana = [[LocalArchiverManager alloc] init];
        });
        return m_localArchiverMana;
    }
    
    - (id)init
    {
        self = [super init];
        if(self)
        {
           self.fileManager = [NSFileManager defaultManager];
        }
        return self;
    }
    
    
    #pragma mark private methods
    
    - (BOOL)checkPathIsExist:(NSString *)path
    {
        return [_fileManager fileExistsAtPath:path isDirectory:nil];
    }
    
    - (void)createArchiverFile
    {
        if (![self checkPathIsExist:ArchiverFile])
        {
            [self addNewFolder:ArchiverFile];
        }
    }
    
    
    //新建目录,path为目录路径(包含目录名)
    - (void)addNewFolder:(NSString *)path
    {
        [_fileManager createDirectoryAtPath:path
                 withIntermediateDirectories:YES
                                  attributes:nil
                                       error:nil];
    }
    
    #pragma mark -
    #pragma mark public methods
    
    - (void)clearArchiverData
    {
        NSError *error;
        if([m_fileManager removeItemAtPath:ArchiverFile error:&error])
        {
        
        }else{
            DLOG(@"清除本地序列化的文件失败....:%@",error);
        }
    }
    
    - (void)saveDataArchiver:(id)obj andAPIKey:(NSString *)key
    {
        NSMutableData *data = [[NSMutableData alloc] init];
        NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
        [archiver encodeObject:obj forKey:key];
        [archiver finishEncoding];
        [self createArchiverFile];
        key = [key stringByReplacingOccurrencesOfString:@"/" withString:@"_"];
        NSString *path = [ArchiverFile stringByAppendingPathComponent:[NSString stringWithFormat:@"%@.text",key]];   
       BOOL isSuc = [data writeToFile:path atomically:YES];
    if(!isSuc) {
            DLOG(@"本地序列化失败key....:%@",key);
        }
    }
    - (id)archiverQueryAPIKey:(NSString *)key
    {
        NSString *str = [key stringByReplacingOccurrencesOfString:@"/" withString:@"_"];
        NSString *path = [ArchiverFile stringByAppendingPathComponent:[NSString stringWithFormat:@"%@.text",str]];
        NSMutableData *data = [[NSMutableData alloc] initWithContentsOfFile:path];
        NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
        id content = [unarchiver decodeObjectForKey:key];
        [unarchiver finishDecoding];
        DLOG(@"content.....:%@",content);
        return content;
    }
  • 相关阅读:
    poj2661Factstone Benchmark
    完整的微信接口类 (转)
    位运算(转载)
    PHP学习笔记之数组游标操作
    MYSQL数据库数据拆分之分库分表总结 (转)
    webservice使用
    MySQL索引类型总结和使用技巧以及注意事项 (转)
    PHP empty、isset、isnull的区别
    myisam和innodb的区别
    看看PHP迭代器的内部执行过程(转)
  • 原文地址:https://www.cnblogs.com/496668219long/p/5635591.html
Copyright © 2020-2023  润新知