iOS中自动实现对象序列化的方法详解

前言

在iOS 中实现对象序列化,需要遵行NSCoding协议,然后对对象的每个属性进行归档和接档赋值,响应的操作比较繁琐。本文主要介绍 利用 runtime遍历属性 大大简化代码量,下面来看看详细的介绍吧。

具体实现代码如下:

1.先建立NSobject的分类, 定义可能用到的相关类型

static NSString *intType  = @"i"; // int_32t(枚举int型)
static NSString *longTpye = @"l"; //long类型
static NSString *longlongType= @"q"; // longlong类型
static NSString *BoolType = @"B"; //bool类型
static NSString *floatType = @"f"; // float
static NSString *doubleType = @"d"; // double
static NSString *boolType = @"c";
static NSString *stringType = @"NSString"; // NSString 类型
static NSString *numberType = @"NSNumber"; // NSNumber 类型
static NSString *arrayType = @"arrayType";//array类型
static NSString *imageType = @"UIImage"; // UIImage 类型

然后在归档方法中便利自身的属性名称,并且取出自身属性对应的值,进行存储到本地。此时遍历类属性本身,用到了Ivar指针(定义对象的实例变量,包括类型和名字),具体代码如下

//归档
- (void)encodeWithCoder:(NSCoder *)aCoder
{
 unsigned int count; // 属性个数
 Ivar *varArray = class_copyIvarList([self class], &count);

 for (int i = 0; i < count; i++)
 {
  Ivar var = varArray[i];
  const char *cName = ivar_getName(var); // 属性名c字符串
  NSString *proName = [[NSString stringWithUTF8String:cName] substringFromIndex:1]; //OC字符串,并且去掉下划线 _
  const char *cType = ivar_getTypeEncoding(var); // 获取变量类型,c字符串

  id value = [self valueForKey:proName];
  NSString *proType = [NSString stringWithUTF8String:cType]; // oc 字符串

  if ([proType containsString:@"NSString"]) {
   proType = stringType;
  }
  if ([proType containsString:@"NSNumber"]) {
   proType = numberType;
  }
  if ([proType containsString:@"NSArray"]) {
   proType = arrayType;
  }
  if ([proType containsString:@"UIImage"]) {
   proType = imageType;
  }

  // (5). 根据类型进行编码
  if ([proType isEqualToString:intType] || [proType isEqualToString:boolType] || [proType isEqualToString:BoolType]) {
   [aCoder encodeInt32:[value intValue] forKey:proName];
  }
  else if ([proType isEqualToString:longTpye]) {
   [aCoder encodeInt64:[value longValue] forKey:proName];
  }
  else if ([proType isEqualToString:floatType]) {
   [aCoder encodeFloat:[value floatValue] forKey:proName];
  }
  else if ([proType isEqualToString:longlongType] || [proType isEqualToString:doubleType]) {
   [aCoder encodeDouble:[value doubleValue] forKey:proName];
  }
  else if ([proType isEqualToString:stringType]) { // string 类型
   [aCoder encodeObject:value forKey:proName];
  }
  else if ([proType isEqualToString:numberType]) {
   [aCoder encodeObject:value forKey:proName];
  }
  else if ([proType isEqualToString:arrayType]) {
   [aCoder encodeObject:value forKey:proName];
  }
  else if ([proType isEqualToString:imageType]) { // image 类型
   [aCoder encodeDataObject:UIImagePNGRepresentation(value)];
  }
 }
 free(varArray);
}

其次进行解档, 原理和归档差不多, 直接上代码

- (instancetype)initWithCoder:(NSCoder *)aDecoder
{
 self = [self init];
 if (self) {

  unsigned int count;
  Ivar *varArray = class_copyIvarList([self class], &count);

  for (int i = 0; i < count; i++) {
   Ivar var = varArray[i];
   const char *cName = ivar_getName(var); // 属性名c字符串
   NSString *proName = [[NSString stringWithUTF8String:cName] substringFromIndex:1]; //OC字符串,并且去掉下划线 _
   const char *cType = ivar_getTypeEncoding(var); // 获取变量类型,c字符串
   NSString *proType = [NSString stringWithUTF8String:cType]; // oc 字符串

   if ([proType containsString:@"NSString"]) {
    proType = stringType;
   }
   if ([proType containsString:@"NSNumber"]) {
    proType = numberType;
   }
   if ([proType containsString:@"NSArray"]) {
    proType = arrayType;
   }
   if ([proType containsString:@"UIImage"]) {
    proType = imageType;
   }

   if ([proType isEqualToString:intType] || [proType isEqualToString:boolType] || [proType isEqualToString:BoolType]) {
    int32_t number = [aDecoder decodeInt32ForKey:proName];
    [self setValue:@(number) forKey:proName];
   }
   else if ([proType isEqualToString:longTpye]) {
    int64_t number = [aDecoder decodeInt64ForKey:proName];
    [self setValue:@(number) forKey:proName];
   }
   else if ([proType isEqualToString:floatType]) {
    float number = [aDecoder decodeFloatForKey:proName];
    [self setValue:@(number) forKey:proName];
   }
   else if ([proType isEqualToString:longlongType] || [proType isEqualToString:doubleType]) {
    double number = [aDecoder decodeFloatForKey:proName];
    [self setValue:@(number) forKey:proName];
   }
   else if ([proType isEqualToString:stringType]) { // string 类型
    NSString *string = [aDecoder decodeObjectForKey:proName];
    [self setValue:string forKey:proName];
   }
   else if ([proType isEqualToString:numberType]) {
    NSString *number = [aDecoder decodeObjectForKey:proName];
    [self setValue:number forKey:proName];
   }
   else if ([proType isEqualToString:arrayType]) {
    NSArray *array = [aDecoder decodeObjectForKey:proName];
    [self setValue:array forKey:proName];
   }
   else if ([proType isEqualToString:imageType]) { // image 类型
    UIImage *image = [UIImage imageWithData:[aDecoder decodeDataObject]];
    [self setValue:image forKey:proName];
   }
  }
 }
 return self;
}

最后也就是 存储方法 、 清除存储的本地缓存 和 获取本地存储数据的方法

//存储路径
- (NSString *)filePathWithUniqueFlagString:(NSString *)uniqueFlag
{
 NSString *docPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
 NSString *detailPath = [NSString stringWithFormat:@"%@_%@",uniqueFlag,[NSString stringWithUTF8String:object_getClassName(self)]];
 NSString *path = [docPath stringByAppendingPathComponent:detailPath];
 return path;
}

//保存对象数据到本地
- (void)saveDataToLocalWithUniqueFlagKey:(NSString *)uniqueFlagKey
{
 [NSKeyedArchiver archiveRootObject:self toFile:[self filePathWithUniqueFlagString:uniqueFlagKey]];
}

//清空本地存储的对象数据
- (id)getDataFromLocalWithUniqueFlagKey:(NSString *)uniqueFlagKey
{
 return [NSKeyedUnarchiver unarchiveObjectWithFile:[self filePathWithUniqueFlagString:uniqueFlagKey]];
}

//从本地获取对象数据
- (BOOL)removeDataFromLocalWithUniqueFlagKey:(NSString *)uniqueFlagKey
{
 NSError *error = nil;
 [[NSFileManager defaultManager] removeItemAtPath:[self filePathWithUniqueFlagString:uniqueFlagKey] error:&error];
 if (!error) {
  return YES;
 }
 else {
  return NO;
 }
}

完整项目下载地址如下:https://github.com/maxzhang123/MXCoding.git  或者可以本地下载地址:http://xiazai.jb51.net/201705/yuanma/MXCoding(jb51.net).rar

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • iOS中自动实现对象序列化的方法详解

    前言 在iOS 中实现对象序列化,需要遵行NSCoding协议,然后对对象的每个属性进行归档和接档赋值,响应的操作比较繁琐.本文主要介绍 利用 runtime遍历属性 大大简化代码量,下面来看看详细的介绍吧. 具体实现代码如下: 1.先建立NSobject的分类, 定义可能用到的相关类型 static NSString *intType = @"i"; // int_32t(枚举int型) static NSString *longTpye = @"l"; //lo

  • jQuery中的deferred对象和extend方法详解

    1 deferred对象 deferred对象是jQuery的回调函数解决方案,它是从jQuery1.5.0版本开始引入的功能 deferred对象的方法 (1) $.Deferred() 生成一个deferred对象. (2) deferred.done() 指定操作成功时的回调函数 (3) deferred.fail() 指定操作失败时的回调函数 (4) deferred.promise() 没有参数时,返回一个新的deferred对象,该对象的运行状态无法被改变:接受参数时,作用为在参数对

  • iOS中封装.framework及使用的方法详解

    .framework是什么? 这个问题相信做iOS的都知道答案. 在我们的日常开发中,经常会用到各种已经封装好的库,比如支付宝.微信SDK等等中的库,这些库可以给我们的开发带来很大的便利.有的时候,由于工作的需要,我们需要对自己的项目进行封装,生成库,方便别人的使用.在这里就边参考好点的博客,边总结一下我们经常看到的.framework. 那什么是"库"呢? "库"是共享程序代码的一种方式!同行总结的这句话很简单也很好的说明了它的作用! 一般的分为"静态库

  • iOS中表情键盘的完整实现方法详解

    前言 最近在公司做了个表情键盘的需求,这个需求的技术难度不会很大,比较偏向业务.但是要把用户体验做的好也是不容易的,其中有几个点需要特别注意.话不多说,下面开始正文(注:本文对应的Demo放在Github上:https://github.com/VernonVan/PPStickerKeyboard (本地上传) ). 市面上的表情键盘的分析 首先来看一下市面上主要的几个APP上的表情键盘,平时使用的时候不会去关注细节,这次特意去使用了表情键盘,发现各个APP的体验还是有优有劣的. 首先是QQ和

  • js基础之DOM中元素对象的属性方法详解

    在 HTML DOM (文档对象模型)中,每个部分都是节点. 节点是DOM结构中最基本的组成单元,每一个HTML标签都是DOM结构的节点. 文档是一个    文档节点 . 所有的HTML元素都是    元素节点 所有 HTML 属性都是    属性节点 文本插入到 HTML 元素是    文本节点 注释是    注释节点. 最基本的节点类型是Node类型,其他所有类型都继承自Node,DOM操作往往是js中开销最大的部分,因而NodeList导致的问题最多.要注意:NodeList是'动态的',

  • JavaScript中Number对象的toFixed() 方法详解

    定义和用法 toFixed() 方法可把 Number 四舍五入为指定小数位数的数字. 语法 NumberObject.toFixed(num) 参数 描述 num 必需.规定小数的位数,是 0 ~ 20 之间的值,包括 0 和 20,有些实现可以支持更大的数值范围.如果省略了该参数,将用 0 代替. 返回值 返回 NumberObject 的字符串表示,不采用指数计数法,小数点后有固定的 num 位数字.如果必要,该数字会被舍入,也可以用 0 补足,以便它达到指定的长度.如果 num 大于 l

  • SpringBoot结合Redis实现序列化的方法详解

    目录 前言 配置类 配置 Jackson2JsonRedisSerializer 序列化策略 配置  RedisTemplate 配置缓存策略 测试代码 完整代码 前言 最近在学习Spring Boot结合Redis时看了一些网上的教程,发现这些教程要么比较老,要么不知道从哪抄得,运行起来有问题.这里分享一下我最新学到的写法 默认情况下,Spring 为我们提供了一个 RedisTemplate 来进行对 Redis 的操作,但是 RedisTemplate 默认配置的是使用Java本机序列化.

  • Spring-IOC容器中的常用注解与使用方法详解

    Spring是什么? Spring是一个轻量级Java开发框架,最早有Rod Johnson创建,目的是为了解决企业级应用开发的业务逻辑层和其他各层的耦合问题.它是一个分层的JavaSE/JavaEE full-stack(一站式)轻量级开源框架,为开发Java应用程序提供全面的基础架构支持.Spring负责基础架构,因此Java开发者可以专注于应用程序的开发. 体系结构 核心容器(Core Container):Spring的核心容器是其他模块建立的基础,有Spring-core.Spring

  • Python自动操作Excel文件的方法详解

    目录 工具 读取Excel文件内容 写入Excel文件内容 Excel文件样式调整 设置表头的位置 设置单元格的宽高 总结 工具 python3.7 Pycharm Excel xlwt&xlrd 读取Excel文件内容 当前文件夹下有一个名为“股票数据.xlsx”的Excel文件,可以按照下列代码方式来操作它. import xlrd # 使用xlrd模块的open_workbook函数打开指定Excel文件并获得Book对象(工作簿) wb = xlrd.open_workbook('股票数

  • python 中Mixin混入类的使用方法详解

    目录 前言 Mixin 与继承的区别 总结 前言 最近在看sanic的源码,发现有很多Mixin的类,大概长成这个样子 class BaseSanic(    RouteMixin,    MiddlewareMixin,    ListenerMixin,    ExceptionMixin,    SignalMixin,    metaclass=SanicMeta, ): 于是对于这种 Mixin 研究了一下,其实也没什么新的东西,Mixin 又称混入,只是一种编程思想的体现,但是在使用

随机推荐