摘要:接上篇源码学习一总览这一次我们来看最新版本的代码,当前最新为在看源码之前,注意这个类。因为它重写了方法,然后把使用的的一些配置,统一写到了这个文件中。项目中还有很多其他的细节就不单说了。
接上篇MJExtension源码学习(一)
总览这一次我们来看MJExtension最新版本的代码,当前最新为3.0.15
在看源码之前,注意MJExtensionConfig这个类。因为它重写了+load方法,然后把使用的model的一些配置,统一写到了这个文件中。
我大致的看了一下代码,随时版本的更新,改变了一些方法的名字和类名,但是其本质的思路是没有变的,跟最初版本一直。然后再看代码的过程中会发现各种判断的情况,还有一些看起来不知道做什么用的代码,其实没关系,因为随着更新这套框架的功能越来越晚上,可以做的事多了,自然为了代码的健壮性就会看到很多类型判断的代码。
我们直接看到核心代码这个方法就可以了
/** 核心代码: */ - (instancetype)mj_setKeyValues:(id)keyValues context:(NSManagedObjectContext *)context
添加了黑名单和白名单的功能,这是获取到响应名单中的成员变量(前提是有设置)
NSArray *allowedPropertyNames = [clazz mj_totalAllowedPropertyNames]; NSArray *ignoredPropertyNames = [clazz mj_totalIgnoredPropertyNames];
往下的mj_enumerateProperties遍历中,返回了一个MJProperty类,他的做用是对每个成员变量做一层封装,代替了最初的MJIvar这个类,MJProperty这个类的内部也更加完善了,这个地方大家可以自己看一下。
再往后就是一些判断然后复制,跟之前的差不多,大家看一下就很容易看明白的。
分析封装MJProperty这里我想说一下MJProperty中下面的这两个方法
/**** 同一个成员属性 - 父类和子类的行为可能不一致(originKey、propertyKeys、objectClassInArray) ****/ /** 设置最原始的key */ - (void)setOriginKey:(id)originKey forClass:(Class)c; /** 对应着字典中的多级key(里面存放的数组,数组里面都是MJPropertyKey对象) */ - (NSArray *)propertyKeysForClass:(Class)c;
但从代码来看很难明确这两个方法到底是做什么用的,我用debug跟了一下,发现其实这个propertyKeysForClass返回的数组中储存的其实是当前成员变量的一个层级关机,我猜测是现在MJExtension可以实现多层级的映射,这样记录下来层级的关系方便后面的映射取值。
下面我们着重看一下在把成员变量封装为MJProperty类的对象时,- (void)setOriginKey:(id)originKey forClass:(Class)c;到底做了什么。
/** * 简单的字典 -> 模型(key替换,比如ID和id。多级映射,比如 oldName 和 name.oldName) */ void keyValues2object4() { // 1.定义一个字典 NSDictionary *dict = @{ @"id" : @"20", @"desciption" : @"好孩子", @"name" : @{ @"newName" : @"lufy", @"oldName" : @"kitty", @"info" : @[ @"test-data", @{@"nameChangedTime" : @"2013-08-07"} ] }, @"other" : @{ @"bag" : @{ @"name" : @"小书包", @"price" : @100.7 } } }; // 2.将字典转为MJStudent模型 MJStudent *stu = [MJStudent mj_objectWithKeyValues:dict];
我们使用这个例子,MJStudent这个类中有下列的成员变量
@property (copy, nonatomic) NSString *ID; @property (copy, nonatomic) NSString *otherName; @property (copy, nonatomic) NSString *nowName; @property (copy, nonatomic) NSString *oldName; @property (copy, nonatomic) NSString *nameChangedTime; @property (copy, nonatomic) NSString *desc; @property (strong, nonatomic) MJBag *bag; @property (strong, nonatomic) NSArray *books;
而且一开始就已经说过了,在MJExtensionConfig中已经实现下面的映射关系
#pragma mark MJStudent中的desc属性对应着字典中的desciption #pragma mark .... [MJStudent mj_setupReplacedKeyFromPropertyName:^NSDictionary *{ return @{ @"desc" : @"desciption", @"oldName" : @"name.oldName", @"nowName" : @"name.newName", @"otherName" : @[@"otherName", @"name.newName", @"name.oldName"], @"nameChangedTime" : @"name.info[1].nameChangedTime", @"bag" : @"other.bag" }; }]; // 相当于在MJStudent.m中实现了+(NSDictionary *)mj_replacedKeyFromPropertyName方法
我们就找一个位置比较深的字段来debug,然后看看她的
[property setOriginKey:[self propertyKey:property.name] forClass:self];到底为他储存了什么数据
我们把断点先打在block的回调中
[clazz mj_enumerateProperties:^(MJProperty *property, BOOL *stop) { }
当遍历到otherName这个成员变量对应的property时,我截下了property中的所有数据,如下图
很容易看出来上面的set方法为_propertyKeysDict存进去一个键值对,key为model的类名,value是一个数组,这是我们只能看到这个数组中又有三个数组,第一个数组有一个元素,第二个两个元素,第三个两个元素。
单从这里看不出这这个_propertyKeysDict种到底存的是什么,如果大家debug跟一下整个set的过程就会看到,整个value大数组有三个元素,是因为我们在映射处理的代码中@"otherName" : @[@"otherName", @"name.newName", @"name.oldName"],otherName对应了一个三个元素的数组,然后从每个元素中看,因为name.newName和name.oldName都是做了一级的映射,所以他们下面的数组会有两个元素,name.newName对应的数组中的两个元素分别name对应的MJPropertyKey和newName对应的MJPropertyKey。
其实到现在,我们已经了解了MJProperty中的成员变量都是为了储存社会,但是就拿这个_propertyKeysDict来说,我们还没有很清楚的明白他为什么要储存一个这样的层级结构,所以我们得继续往下看后面的取值和赋值。
这个地方的就很简单了,其实还是为了多级映射的取值
// 1.取出属性值 id value; NSArray *propertyKeyses = [property propertyKeysForClass:clazz]; for (NSArray *propertyKeys in propertyKeyses) { value = keyValues; for (MJPropertyKey *propertyKey in propertyKeys) { value = [propertyKey valueInObject:value]; } if (value) break; }
其中嵌套的for循环,就可以一层层的找到最深的那个映射,就拿刚才的name.newName来说,以为封装了两个MJPropertyKey,在遍历第一次的时候,拿到了name对应的数据,然后在遍历第二次拿到newName的数据,可以看出来,每一次的外层的遍历过程中,如果如果取到value值,遍历就结束了, 所以咱们用的例子最终打印的结果otherName=lufy,,也就是name.newName的值。
个人感觉这种写成对应数组的映射,可能是处理有些映射取不到值,哪个有值用哪个 ~ 如果是单纯的多级映射,其实@"oldName" : @"name.oldName",这种写法就是ok的。
项目中还有很多其他的细节就不单说了。
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/17069.html
摘要:继续进行优秀开源框架的源码学习,这次打算学习一些常用的解析的框架,比如,,等。这句代码也就是递归思想的提现,如果这个成员变量是一个自定义类的,那么该成员变量对应的值应该也是一个,所以用这个二级的类继续调用方法,继续解析下去。 继续进行优秀开源框架的源码学习,这次打算学习一些常用的model解析的框架,比如YYModel,MJExtension,Mantle等。我自己用过YYModel和...
摘要:简单来说,深拷贝就是内容拷贝,浅拷贝就是指针拷贝。更具体的用法,就不作过多解释,自行百度吧。由于数据的复杂性或者没有实现协议,简单的使用来拷贝可能无法达到预期效果,由此就有了的衍生使用。 首先了解两个概念:深拷贝和浅拷贝。简单来说,深拷贝就是内容拷贝,浅拷贝就是指针拷贝。更具体的用法,就不作过多解释,自行百度吧。 // 字符串是直接赋值的,右侧如果是copy,那么就是浅拷贝;右侧如果是...
摘要:交代一下背景最开始是因为公司要求写单元测试。写单元测试是一件比较痛苦的事情,尤其是在项目已经成型之后。 看了十来篇关于MVVM的文章之后,终于开始有信心在自己的项目中尝试采用MVVM这个架构了。 交代一下背景 最开始是因为公司要求写单元测试。写单元测试是一件比较痛苦的事情,尤其是在项目已经成型之后。懒惰驱使我必须去了解有没有更具吸引力的替代方式,碰巧看到一篇关于MVVM的文章,讲到MV...
摘要:一添加控件添加控件使用约束,控制底部文字的位置和高度。增加需要的数据这里,我们将图片和文字保存在中创建该控件的模型数据绑定无限滚动年控制器无限滚动年注册 一、添加控件 (1)、添加xib控件showImg(https://segmentfault.com/img/bVzYlx); 使用约束,控制底部文字text的位置和高度。 (2)、增加需要的数据这里,我们将图片和文字保存在plist...
阅读 2978·2023-04-26 03:05
阅读 3627·2021-11-15 11:37
阅读 1211·2021-10-13 09:39
阅读 2808·2021-09-10 10:50
阅读 1353·2019-08-30 13:09
阅读 1800·2019-08-30 13:05
阅读 715·2019-08-29 12:42
阅读 1278·2019-08-28 18:18