当前位置: 首页 > news >正文

iOS开发之遍历Model类的属性并完善使用Runtime给Model类赋值

在上篇博客iOS开发之使用Runtime给Model类赋值》中介绍了如何使用运行时在实体类的基类中添加给实体类的属性赋值的方法,这个方法的前提是字典的Key必须和实体类的Property Name相同,然后通过运行时来生成和执行Setter方法给Model类的属性赋值。

  通过Runtime来给Model类属性赋值的好处是多多的,它便于代码的后期维护,并且提高了开发效率。当你拿到解析后的字典时你不用一个一个的通过key去把字典的值赋值给相应的Model类的属性,本篇博客中会给出如何去遍历Model中属性的值,并且给出字典的Key和Model的属性名不一样的情况我们该如何负值。

  接下来会在上一个博客代码基础上在Model基类中添加通过Runtime来遍历Model类的属性值。

  一、获取Model的实体属性

  1.要想遍历Model类的属性,首先得通过Runtime来获取该Model类有哪些属性,输出Model的所有属性的值可不像遍历Dictionary和Array那样一个for循环搞定的,下面的方法是通过Runtime来获取Model类的属性字符串,并以数组的形式返回。代码如下:

///通过运行时获取当前对象的所有属性的名称,以数组的形式返回
- (NSArray *) allPropertyNames{
 ///存储所有的属性名称
 NSMutableArray *allNames = [[NSMutableArray alloc] init];
 
 ///存储属性的个数
 unsigned int propertyCount = 0;
 
 ///通过运行时获取当前类的属性
 objc_property_t *propertys = class_copyPropertyList([self class], &propertyCount);
 
 //把属性放到数组中
 for (int i = 0; i < propertyCount; i ++) {
 ///取出第一个属性
 objc_property_t property = propertys[i];
 
 const char * propertyName = property_getName(property);
 
 [allNames addObject:[NSString stringWithUTF8String:propertyName]];
 }
 
 ///释放
 free(propertys);
 
 return allNames;
}

  2.获取到Model类的属性方法后需要把属性字符串生成get方法,我们可以执行get方法来获取Model属性的值,下方的方法是根据属性字符串来获取属性的getter方法,OC中属性的getter方法的名字和属性的名字是一致的,生成getter方法比较简单,具体代码如下:

#pragma mark -- 通过字符串来创建该字符串的Setter方法,并返回
- (SEL) creatGetterWithPropertyName: (NSString *) propertyName{
 
 //1.返回get方法: oc中的get方法就是属性的本身
 return NSSelectorFromString(propertyName);
}

  二、Get方法的执行

    接下来要做的是通过Runtime来执行Getter方法,这一块需要通过方法的签名来执行Getter方法。在OC的运行时中要执行的方法需要传入参数或者需要接收返回值时,需要通过方法的签名来调用方法。下面的代码就是创建方法的签名,然后通过签名来获取调用的对象,在下边的方中回调用上述两个方法在通过方法的签名来获取Model属性的值,具体代码如下:

- (void) displayCurrentModleProperty{
 
 //获取实体类的属性名
 NSArray *array = [self allPropertyNames];
 
 //拼接参数
 NSMutableString *resultString = [[NSMutableString alloc] init];
 
 for (int i = 0; i < array.count; i ++) {
 
 //获取get方法
 SEL getSel = [self creatGetterWithPropertyName:array[i]];
 
 if ([self respondsToSelector:getSel]) {
 
 //获得类和方法的签名
 NSMethodSignature *signature = [self methodSignatureForSelector:getSel];

 //从签名获得调用对象
 NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];

 //设置target
 [invocation setTarget:self];
 
 //设置selector
 [invocation setSelector:getSel];

 //接收返回的值
 NSObject *__unsafe_unretained returnValue = nil;

 //调用
 [invocation invoke];

 //接收返回值
 [invocation getReturnValue:&returnValue];

 [resultString appendFormat:@"%@\n", returnValue];
 }
 }
 NSLog(@"%@", resultString);
 
}


  执行上述方法就可以输入Model中的属性的值,下面就在main函数中对Model赋完值后调用上述方法输出一下Model的属性值,调用代码如下所示:
1 BeautifulGirlModel *beautifulGirl = [BeautifulGirlModel modelWithDictionary:data];
2 
3 [beautifulGirl displayCurrentModleProperty];

  运行结果如下,下面的输出结果是Model中属性的值。

  三、Dictionary的Key与Model的属性不同的处理方式

    有时候会遇到字典的key和Model的属性不一样的情况,那么如何去解决这个问题呢?最简单的做法是在具体的实体类中去维护一个映射关系方法,通过这个方法我们可以获取相应的的映射关系。

    1.在Model的基类中添加一个返回映射字典的一个方法,然后在子类中进行重写,这个映射方法在基类中返回nil, 如果子类需要重写的话就对这个方法进行重写并返回映射字典。方法如下:

1 #pragma 返回属性和字典key的映射关系
2 -(NSDictionary *) propertyMapDic{
3 return nil;
4 }

  2.修改一下我们的便利初始化方法,在有映射字典的情况和没有映射字典的情况下调用的方法是不一样的,便利初始化方法的代码如下:

- (instancetype)initWithDictionary: (NSDictionary *) data{
 {
 self = [super init];
 if (self) {
 if ([self propertyMapDic] == nil) {
 [self assginToPropertyWithDictionary:data];
 } else {
 [self assginToPropertyWithNoMapDictionary:data];
 }
 }
 return self;
 }
}

  3.接下来就将实现有映射关系要调用的方法,这个方法就是通过映射关系把字典的key转换成与property的名字一样的字典,然后调用之前的赋值方法,具体代码如下:

#pragma 根据映射关系来给Model的属性赋值
-(void) assginToPropertyWithNoMapDictionary: (NSDictionary *) data{
 ///获取字典和Model属性的映射关系
 NSDictionary *propertyMapDic = [self propertyMapDic];
 
 ///转化成key和property一样的字典,然后调用assginToPropertyWithDictionary方法
 
 NSArray *dicKey = [data allKeys];

 
 NSMutableDictionary *tempDic = [[NSMutableDictionary alloc] initWithCapacity:dicKey.count];
 
 for (int i = 0; i < dicKey.count; i ++) {
 NSString *key = dicKey[i];
 [tempDic setObject:data[key] forKey:propertyMapDic[key]];
 }
 
 [self assginToPropertyWithDictionary:tempDic];
 
}

  4.创建一个BadBoyModel, 并重写propertyMapDic方法,并且在propertyMapDic方法中给出映射关系并返回该映射关系对应的字典。

    (1)BadBoyModel的属性如下:

//
// BadBoyModel.h
// BaseModelProject
//
// Created by Mr.LuDashi on 15/7/24.
// Copyright (c) 2015年 ludashi. All rights reserved.
//

#import "BaseModelObject.h"

@interface BadBoyModel : BaseModelObject

@property (nonatomic, copy) NSString *boy1;
@property (nonatomic, copy) NSString *boy2;
@property (nonatomic, copy) NSString *boy3;
@property (nonatomic, copy) NSString *boy4;

@end

    (2)重写映射方法,映射字典的key是要转换字典的key, Value是对应Model的属性名。

//
// BadBoyModel.m
// BaseModelProject
//
// Created by Mr.LuDashi on 15/7/24.
// Copyright (c) 2015年 ludashi. All rights reserved.
//

#import "BadBoyModel.h"

@implementation BadBoyModel

#pragma 返回属性和字典key的映射关系
-(NSDictionary *) propertyMapDic{
 return @{@"keyBoy1":@"boy1",
 @"keyBoy2":@"boy2",
 @"keyBoy3":@"boy3",
 @"keyBoy4":@"boy4",};
}

@end


5.在main函数中进行测试

      (1)、生成我们的数值字典,字典的key与要赋值Model的属性不同,下面的循环就是要生成测试使用的数据:

//生成Dic的Key与Model的属性不一样的字典。
 
 NSMutableDictionary *data1 = [[NSMutableDictionary alloc] init];
 
 //创建测试适用的字典
 for(int i = 1; i <= 4; i ++){
 NSString *key = [NSString stringWithFormat:@"keyBoy%d", i];
 
 NSString *value = [NSString stringWithFormat:@"我是第%d个坏男孩", i];
 
 [data1 setObject:value forKey:key];
 }

      (2) 实例化Model并输出结果,当然之前的代码也是可以使用的。

1 BadBoyModel *badBoyModel = [BadBoyModel modelWithDictionary:data1];
2 
3 [badBoyModel displayCurrentModleProperty];

    运行输出结果如下:

    今天博客就到这,至此,Model的基类最基本的方法封装的也就差不多了,根据具体需求可以在添加新的方法

相关文章:

  • html中块注释!--[if IE]….![endif]-- (!--[if !IE]||![endif]
  • 在CentOS7部署zookeeper集群以及简单API使用
  • Spring3.0核心组件的源码简单分析
  • 核心动画——弹簧动画二
  • 不能修改数据
  • 春招面试小记
  • poj 3468(线段树+lazy思想)
  • 算法系列15天速成——第五天 五大经典查找【中】
  • Mac OS 安装phpMyAdmin
  • 用PHP抓取淘宝商品的用户晒单评论+图片实例
  • c语言小练习(蛮好玩的)
  • Android上传头像代码,相机,相册,裁剪
  • [ Algorithm ] N次方算法 N Square 动态规划解决
  • Memcached 安装及启动脚本
  • kafka 源码调研系列1 特色
  • hexo+github搭建个人博客
  • IE9 : DOM Exception: INVALID_CHARACTER_ERR (5)
  • 【跃迁之路】【444天】程序员高效学习方法论探索系列(实验阶段201-2018.04.25)...
  • AngularJS指令开发(1)——参数详解
  • css系列之关于字体的事
  • Docker下部署自己的LNMP工作环境
  • ECMAScript入门(七)--Module语法
  • Java 网络编程(2):UDP 的使用
  • JSONP原理
  • nginx 负载服务器优化
  • Python学习之路16-使用API
  • Swoft 源码剖析 - 代码自动更新机制
  • UEditor初始化失败(实例已存在,但视图未渲染出来,单页化)
  • vue2.0项目引入element-ui
  • webpack项目中使用grunt监听文件变动自动打包编译
  • win10下安装mysql5.7
  • 离散点最小(凸)包围边界查找
  • 前端每日实战:61# 视频演示如何用纯 CSS 创作一只咖啡壶
  • 前端面试之CSS3新特性
  • 通过来模仿稀土掘金个人页面的布局来学习使用CoordinatorLayout
  • 我这样减少了26.5M Java内存!
  • 用Node EJS写一个爬虫脚本每天定时给心爱的她发一封暖心邮件
  • 阿里云ACE认证学习知识点梳理
  • #《AI中文版》V3 第 1 章 概述
  • #LLM入门|Prompt#1.7_文本拓展_Expanding
  • #我与虚拟机的故事#连载20:周志明虚拟机第 3 版:到底值不值得买?
  • (二)windows配置JDK环境
  • (三)elasticsearch 源码之启动流程分析
  • (十三)Flask之特殊装饰器详解
  • (一)Mocha源码阅读: 项目结构及命令行启动
  • (转)关于如何学好游戏3D引擎编程的一些经验
  • .NET CLR Hosting 简介
  • .net 程序 换成 java,NET程序员如何转行为J2EE之java基础上(9)
  • .NET 反射的使用
  • .NET/C# 利用 Walterlv.WeakEvents 高性能地定义和使用弱事件
  • .Net的DataSet直接与SQL2005交互
  • .net开发引用程序集提示没有强名称的解决办法
  • .NET学习教程二——.net基础定义+VS常用设置
  • .Net组件程序设计之线程、并发管理(一)
  • @private @protected @public