iPhoneSDK开发136技系列:第23技使用NSCoding保存数据

本系列文章基于日本最近热卖的《PhoneSDK开发136技》一书的目录和实例代码。由于无法获得此书,因此只能猜测加个人发挥。确切地说应该是一个读码笔记系列。

     为了将应用数据存储到硬盘中,iOS提供基本的文件API、Property List序列化、SQLite、CoreData以及NSCoding。对于轻量级的数据要求,NSCoding因其简单而成为一种比较合适的方式。
     NSCoding是一个你需要在数据类上要实现的协议以支持数据类和数据流间的编码和解码。数据流可以持久化到硬盘。而实现NSCoding其实很简单,因此有时会很有用。相关代码如下:
 

  1. // 键值宏定义  
  2. #define SAMPLEDATA_KEY_STARS (@"stars")  
  3. #define SAMPLEDATA_KEY_TITLE (@"title")  
  4. #define SAMPLEDATA_KEY_ITEMS (@"items")  
  5.  
  6. @interface SampleData : NSObject <NSCoding>{  
  7.     int starCount;  
  8.     NSString *titleText;  
  9.     NSMutableDictionary *items;  
  10.     int tempNumber; //不保存的变量  
  11. }  
  12. @property(nonatomic,readwrite) int starCount;  
  13. @property (nonatomic, retain, readwrite) NSString *titleText;  
  14. @property (nonatomic, retain, readwrite) NSMutableDictionary *items;  
  15. @property(nonatomic,readwrite) int tempNumber;  
  16. @end  
  17.  
  18. @implementation SampleData  
  19. @synthesize starCount, titleText, items, tempNumber;  
  20.  
  21. - (void)encodeWithCoder:(NSCoder*)coder  
  22. {  
  23.     [coder encodeObject:titleText forKey:SAMPLEDATA_KEY_TITLE];  
  24.     [coder encodeObject:items forKey:SAMPLEDATA_KEY_ITEMS];  
  25.     [coder encodeInt:starCount forKey:SAMPLEDATA_KEY_STARS];  
  26. }  
  27. - (id)initWithCoder:(NSCoder*)decoder  
  28. {  
  29.     if ((self = [super init])){  
  30.         //被init:调用时使用默认值  
  31.         self.titleText = nil;  
  32.         self.items = [[NSMutableDictionary alloc]init];  
  33.         starCount = 0;  
  34.         tempNumber = 0;  
  35.         if (decoder==nil) {  
  36.             return self;  
  37.         }  
  38.  
  39.         // 取值前确认相应的键值是否存在。  
  40.         if ([decoder containsValueForKey:SAMPLEDATA_KEY_TITLE]) {  
  41.             self.titleText =  
  42.                 [decoder decodeObjectForKey:SAMPLEDATA_KEY_TITLE];  
  43.         }  
  44.         if ([decoder containsValueForKey:SAMPLEDATA_KEY_ITEMS]) {  
  45.             self.items =  
  46.                 [decoder decodeObjectForKey:SAMPLEDATA_KEY_ITEMS];  
  47.         }  
  48.         if ([decoder containsValueForKey:SAMPLEDATA_KEY_STARS]) {  
  49.             starCount = [decoder decodeIntForKey:SAMPLEDATA_KEY_STARS];  
  50.         }  
  51.     }  
  52.     return self;  
  53. }  
  54. - (id)init {  
  55.     if ((self = [super init])){  
  56.         [self initWithCoder:nil];  
  57.     }  
  58.     return self;  
  59. }  
  60. @end 

     就这些! 其中我们需要实现两个方法: encodeWithCoder和initWithEncoder。encodeWithCoder就是编码,initWithCoder就是解码。encodeWithCoder方法传入的是一个NSCoder对象,实现的时候我们就可以调用encodeObject、encodeFloat、encodeInt等各种方法并通过指定键值进行编码。