zoukankan      html  css  js  c++  java
  • iphone开发中数据持久化之——模型对象归档(二)

             在Cocoa世界中,术语“归档”是指另一种形式的序列化,它可以实现对任何对象的序列化。使用对模型对象进行归档的技术可以轻松将复杂的对象写入文件,然后再从中读取它们。只要在类中实现的每个属性都是标量(如int或float)或都是符合NSCoding协议的某个类的实例,就可以对该对象进行归档。由于大多数支持数据存储的Foundation和Cocoa Touch类都符合NSCoding,因此对于大多数类来说,归档相对而言比较容易实现。为了增加灵活性,在使用应用中,NSCoding协议最好与NSCopying协议一起使用,该协议运行复制对象。

              1)符合NSCoding

              NSCoding协议申明了两个方法,这两个方法都是必需的。一个方法将对象编码到归档中;另一个方法通过对归档解码来创建一个新的对象。这两个方法都传递一个NSCoder实例,我们继续根据以上一篇博文中的属性序列化中的例子为基础来说明。首先,我们创建一个FourLines类,该类的头文件如下代码如下,主要申明几个属性。

    #import <Foundation/Foundation.h>
    #define kField1Key @"field1"
    #define kField2Key @"field2"
    #define kField3Key @"field3"
    #define kField4Key @"field4"
    
    @interface FourLines : NSObject<NSCoding,NSCopying>
    {
        NSString* field1;
        NSString* field2;
        NSString* field3;
        NSString* field4;
    }
    
    @property (nonatomic, retain) NSString* field1;
    @property (nonatomic, retain) NSString* field2;
    @property (nonatomic, retain) NSString* field3;
    @property (nonatomic, retain) NSString* field4;

             接下来,需要实现 NSCoding协议的第一个方法encodeWithCoder,该方法主要是对对象进行编码,代码如下:

    // NSCoding
    - (void)encodeWithCoder:(NSCoder *)aCoder
    {
        [aCoder encodeObject:field1 forKey:kField1Key];
        [aCoder encodeObject:field2 forKey:kField2Key];
        [aCoder encodeObject:field3 forKey:kField3Key];
        [aCoder encodeObject:field4 forKey:kField4Key];
    }

            上面代码将对象的每个实例变量进行了编码,因此,在对归档数据进行操作时,我们需要对编码的对象进行解码,因此 NSCoding的另一个方法就是实现对对象的解码功能,代码如下:

    - (id)initWithCoder:(NSCoder *)aDecoder
    {
        if (self = [super init])
        {
            self.field1 = [aDecoder decodeObjectForKey:kField1Key];
            self.field2 = [aDecoder decodeObjectForKey:kField2Key];
            self.field3 = [aDecoder decodeObjectForKey:kField3Key];
            self.field4 = [aDecoder decodeObjectForKey:kField4Key];
        }
        return self;
    }

              2)实现NSCopying        

              实现了NSCoding协议的两个方法,接下来我们还要实现NSCopying协议中的copyWithZone方法,该方法可用于复制对象,与NSCoding中的initWithCoder很相似,所有需要做的只是创建一个同类对象,然后将该新实例的所有属性都设置为与该对象属性相同的只。该方法的代码如下:

    // NSCopying
    - (id)copyWithZone:(NSZone *)zone
    {
        FourLines* copy = [[[self class] allocWithZone:zone]init];
        field1 = [self.field1 copy];
        field2 = [self.field2 copy];
        field3 = [self.field3 copy];
        field4 = [self.field4 copy];
        return copy;
    }

              3)对数据对象进行归档

               从符合NSCoding的一个或多个对象创建归档相对比较容易,首先创建一个NSMutableData实例,用于包含编码的数据;然后创建一个NSKeyedArchiver实例,用于将对象归档到NSMutableData实例中。在这两个对象创建之后,就直接告知归档程序,将NSMutableData写入文件系统。本文在程序即将退出时,对数据进行归档,代码如下:

    - (void)applicationWillTerminate:(NSNotification *)notification
    {
    
        FourLines* fourLines = [[FourLines alloc]init];
        fourLines.field1 = field1.text;
        fourLines.field2 = field2.text;
        fourLines.field3 = field3.text;
        fourLines.field4 = field4.text;
        
        NSMutableData* data = [[NSMutableData alloc]init];
        NSKeyedArchiver* archiver = [[NSKeyedArchiver alloc]initForWritingWithMutableData:data];
        [archiver encodeObject:fourLines forKey:kDataKey];
        [archiver finishEncoding];
        [data writeToFile:[self dataFilePath] atomically:YES];
    }

            4)从归档文件中加载数据

             在程序重新启动或需要再次加载归档对象时,我们只要创建一个NSData实例,并创建一个NSKeyedUnarchiver对象对数据进行解码,该代码在viewDidLoad函数中实现,代码如下:

    - (void)viewDidLoad
    {
    	// Do any additional setup after loading the view, typically from a nib.
        NSString* filePath = [self dataFilePath];
        if ([[NSFileManager defaultManager] fileExistsAtPath:filePath])
        { 
            NSData* data = [[NSMutableData alloc]initWithContentsOfFile:[self dataFilePath]];
            NSKeyedUnarchiver* unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
            FourLines* fourLines = [unarchiver decodeObjectForKey:kDataKey];
            [unarchiver finishDecoding];
            field1.text = fourLines.field1;
            field2.text = fourLines.field2;
            field3.text = fourLines.field3;
            field4.text = fourLines.field4;
        }
        
        UIApplication* app = [UIApplication sharedApplication];
        [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationWillTerminate:) name:UIApplicationWillTerminateNotification object:app];
        
       [super viewDidLoad];
    }

             程序实现的功能与上一篇博文相同,实际上,这两个程序大致相似,改动的地方并不多。那么属性列表和对象归档又有什么优势呢?举个简单的例子:实现比较int , char, float ,double两个变量的大小,用C语言要些4个函数,在C++中有函数模板,对于实现类似功能的函数,通过模板实现,只需要一个函数就可以完成相同功能,模板函数不仅减少了代码编写量,而且提高了程序的可维护性,函数模板就类似这里的对象归档。

            该实例的代码可以通过下面的连接下载:http://download.csdn.net/detail/kezunhai/6199397









  • 相关阅读:
    编译KlayGE所需要的第三方库和工具下载
    KlayGE启用顶级域名
    Sophus和Eigen 李群李代数 简单介绍
    G2O曲线拟合1
    梯度下降
    PCL1.8单张图点云转换显示
    双目测距demo
    Kinect基于微软SDK彩图与深度图对齐
    单例模式
    zendstudio卡死
  • 原文地址:https://www.cnblogs.com/pangblog/p/3299403.html
Copyright © 2011-2022 走看看