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的属性值,调用代码如下所示:

BeautifulGirlModel *beautifulGirl = [BeautifulGirlModel modelWithDictionary:data];

[beautifulGirl displayCurrentModleProperty];

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

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

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

#pragma 返回属性和字典key的映射关系

-(NSDictionary *) propertyMapDic{

return nil;

}

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并输出结果,当然之前的代码也是可以使用的。

BadBoyModel *badBoyModel = [BadBoyModel modelWithDictionary:data1];

[badBoyModel displayCurrentModleProperty];

运行输出结果如下:

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

作者:青玉伏案

出处:http://www.cnblogs.com/ludashi/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

如果文中有什么错误,欢迎指出。以免更多的人被误导。

时间: 2024-10-18 19:19:57

iOS开发>学无止境 - 遍历Model类的属性并完善使用Runtime给Model类赋值的相关文章

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

http://www.cocoachina.com/ios/20150807/12877.html 在上篇博客<iOS开发之使用Runtime给Model类赋值>中介绍了如何使用运行时在实体类的基类中添加给实体类的属性赋值的方法,这个方法的前提是字典的Key必须和实体类的Property Name相同,然后通过运行时来生成和执行Setter方法给Model类的属性赋值. 通过Runtime来给Model类属性赋值的好处是多多的,它便于代码的后期维护,并且提高了开发效率.当你拿到解析后的字典时你

iOS开发UI篇—CAlayer层的属性(转摘)

iOS开发UI篇—CAlayer层的属性 一.position和anchorPoint 1.简单介绍 CALayer有2个非常重要的属性:position和anchorPoint @property CGPoint position; 用来设置CALayer在父层中的位置 以父层的左上角为原点(0, 0) @property CGPoint anchorPoint; 称为“定位点”.“锚点” 决定着CALayer身上的哪个点会在position属性所指的位置 以自己的左上角为原点(0, 0) 它

iOS开发UI篇—无限轮播(功能完善)

iOS开发UI篇—无限轮播(功能完善) 一.自动滚动 添加并设置一个定时器,每个2.0秒,就跳转到下一条. 获取当前正在展示的位置. 1 [self addNSTimer]; 2 } 3 4 -(void)addNSTimer 5 { 6 // NSTimer timerWithTimeInterval:<#(NSTimeInterval)#> target:<#(id)#> selector:<#(SEL)#> userInfo:<#(id)#> repe

iOS开发&gt;学无止境 - 浅谈MVVM的架构设计与团队协作

李刚按:本文是青玉伏案写的一篇文章.相信大家对MVC耳熟能详,MVVM可能听说的相对少一些,这一篇文章将会想你阐述MVVM设计,还有团队协作的经验分享.如果你也觉得不错,就分享一下吧! demo:https://github.com/lizelu/MVVM 今天写这篇文章是想达到抛砖引玉的作用,想与大家交流一下思想,相互学习,博文中有不足之处还望大家批评指正.本篇文章的内容沿袭以往博客的风格,也是以干货为主,偶尔扯扯咸蛋(哈哈~不好好工作又开始发表博客啦~). 由 于本人项目经验有限,关于架构设

iOS开发&gt;学无止境 - 沙盒和文件操作(汇总)

关于沙盒和文件操作的文章写了几篇,在写的过程中加深了自己的记忆.今天来做一下汇总,算是对知识点的梳理. 文章梳理 1 iOS开发之沙盒机制(SandBox) 这是一篇关于沙盒的基础知识教程.简述沙盒的作用,对Documents.Library.tmp之间的区别做了介绍.通过两种方法打开沙盒,查看其中的内容. 2 iOS开发之获取沙盒路径 沙盒里的文件夹包括Documents.Library.tmp.文章介绍了如何获取Documents.Library.Caches.tmp的路径. 3 如何查看真

iOS开发&gt;学无止境 - 保存照片到自己创建的相簿

在刚刚在线上一篇文章iOS开发之保存照片到系统相册(Photo Album),我们讲到了如何保持照片系统相册.还有其他保存的方法吗? 保存照片还可以用ALAssetsLibrary,ALAssetsLibrary提供了我们对iOS设备中的相片.视频的访问,是连接应用程序和相册之间访问的一个桥梁. 接下来,我们来详细讲解一下关于系统相册权限获取.保存照片.创建自己的相簿等等功能. 创建自己的相簿 这也是一种比较创建的作法,创建自己的相簿,然后把照片或者视频保存到自己的相簿中.相关代码如下: ALA

iOS开发&gt;学无止境 - UITablView上下滑动控制底部按钮的出现和消失

今天我们一起来做一个好玩的功能,通过UITablView上下滑动控制底部按钮的出现和消失.先来看看效果: 需求分析 要做一个这样的功能,对你来说应该不难,实现的方式也有很多.我们来分析一下要实现那几个小功能点: UITablView向上滑动,底部按钮消失 UITablView向下滑动,底部按钮出现 UITablView滑动到底部,底部按钮出现 主要就是这三个小功能点.那么很简单,我们只需要判断UITablView是往上滑动,还是往下滑动,以及判断UITablView是否滑动到底部即可. 代码实现

iOS开发——swift篇&amp;Swift新特性(二)函数、枚举、类与结构

函数.枚举.类与结构 可变参数 Swift的函数可以接受零个或多个指定类型的参数值,使用...来表示传递的是可变参数 func arithmeticMean(numbers: Double...) -> Double { var total: Double = 0 for number in numbers { total += number } return total / Double(numbers.count) } arithmeticMean(1, 2, 3, 4, 5) // ret

iOS开发&gt;学无止境 - 使用MVC模式帮ViewController瘦身

随着程序逻辑复杂度的提高,你是否也发现了App中一些ViewController的代码行数急剧增多,达到了2,3千行,甚至更多.这时如果想再添加一点功能或者修改现有逻辑变得让人无比头疼.如果你遇到了这类问题,那是时候停下来了,思考一下如何更好地组织代码,给VC瘦身.本文将会阐述如何结合MVC的思想帮你的VC瘦身同时提高复用和可扩展性. 一.开发中常见的现象和缺点 iOS中最常见的一种设计模式就是MVC,但在实际开发过程中,我们因为这样.那样的原因让单纯的ViewController变成了集Mod