Mantle是什么?
Mantle makes it easy to write a simple model layer for your Cocoa or Cocoa Touch application. --Mantle是构建Cocoa或者Cocoa Touch应用程序的Model层的框架。
Mantle地址:GitHub - Mantle/Mantle: Model framework for Cocoa and Cocoa Touch
Xtrace是什么?
Xtrace是ios中强大的调试的库,能详细打印出一个某个方法被调用的堆栈,方便调试时定位问题。
Xtrace地址:GitHub - johnno1962/Xtrace: Trace Objective-C method calls by class or instance
OK~提供了以上信息之后,正式开始我们的主题。不同于其他大神模式,一上来就分析源码+总结,今天我们借助与Xtrace工具来一步步教你读懂Mantle源码。
开始
在我们的Mantle案例中需要添加Xtrace的支持,将Xtrace.h和Xtrace.mm两个文件添加到我们的项目中,并在AppDelegate.m中添加Xtrace的引用和配置相关项。
大家了解过Mantle使用的人大致应该了解MTLModel类、MTLJSONAdapter类以及MTLJSONSerializing协议是整个Mantle的核心,所以我们在上面的XTrace配置项中针对MTLModel的子类GHIssueMantle(实现MTLJSONSerializing协议)和MTLJSONAdapter类进行堆栈跟踪。以下是我们解析过程中经常使用的代码:
我们来看一下XTrace跟踪到的堆栈情况:
大家可以看到从入口代码到Mantle内部的执行过程就非常的清晰了,整个过程从JSON串转换到Model所涉及到的方法的执行过程是按顺序进行的,所以接下来我们主要针对这些这些方法进行一个解读。
方法一:modelOfClass:fromJSONDictionary:error:
这个方法通过initWithModelClass方法初始化MTLJSONAdapter实例,之后调用modelFromJSONDictionary: error:将JSON字典转化成Model;
方法二:initWithModelClass:
这个方法会主要用来初始化变量,并返回MTLJSONAdapter实例。先是调用了JSONKeyPathsByPropertyKey来指定对象的属性如何映射到不同的key path,并将结果保存。再是调用propertyKeys方法获取model中所有属性(除了存储类型为MTLPropertyStorageNone)。接着遍历_JSONKeyPathsByPropertyKey变量,先判断是不是包含在propertyKeys集合中,如果未包含直接return掉;如果包含则获取对应属性keyPath,这里keyPath有两种形式,一种字符串的形式、一种数组的形式,所以这里做了一下判断。然后调用了valueTransformersForModelClass方法来获取model中每个属性对应的值转换器的字典。最后初始化了JSONAdaptersByModelClass变量并返回实例。
方法三:JSONKeyPathsByPropertyKey
这是子类Model遵守MTLJSONSerializing协议必须实现的方法,主要指定对象的属性如何映射到不同的key path。
方法四:propertyKeys
这个方法主要获取获取model中所有属性(除了存储类型为MTLPropertyStorageNone)。这里通过关联对象形式来获取缓存,如果缓存为Nil,则调用enumeratePropertiesUsingBlock一次遍历所有属性,通过调用storageBehaviorForPropertyWithKey获取每个属性的存储行为,并将存储行为不是MTLPropertyStorageNone的结果加入到集合中,最后结果关联到当前对象并返回。
方法五:enumeratePropertiesUsingBlock
这个方法主要来循环获取model及其父类(非MTLModel)的所有属性,并执行block。这里运用到了runtime函数class_copyPropertyList来获取属性列表。
方法六:storageBehaviorForPropertyWithKey
这个方法主要用来获取指定属性的存储行为。先是通过runtime获取属性,并通过mtl_copyPropertyAttributes获取属性内部结构。之后的判断逻辑相对比较复杂,同时也说明作者考虑比较周到。从这个逻辑判断我们可以知道,Mantle实际将存储类型分类为两种(可存储和不可存储),而不可存储的属性是没有变量空间ivar,readonly属性往往会是这种情况。这里内部出现嵌套调用的情况,主要是考虑到了一个被覆盖的readonly属性其父类可能存在变量空间。
方法七:valueTransformersForModelClass
这个方法用来获取model中属性对应的值转换器集合。先去判断了是否实现"属性名称"+"JSONTransformer"方法,如果实现进行动态调用并返回。如果未实现则判断是否实现JSONTransformerForKey这个方法,如果实现则进行调用并返回。如果未实现,则对属性结构进行判断,返回合适的值转换器。这里因为我们实现了URLJSONTransformer这个方法,所以代用完URLJSONTransformer就直接返回了。
注意:第373行的函数调用方式可以在一个循环内频繁地调用一个特定的方法时,通过这种方式可以提高程序的性能,免去了runtime方法查找的性能损失。
方法八:modelOfClass:fromJSONDictionary:error:
这个方法是将JSON转换为Model最主要的方法,前面做的都是一些初始化工作。先是判断当前类是否实现了classForParsingJSONDictionary方法,这个方法在存在类族的情况下判断具体哪个类来执行转换工作。接着遍历model的所有属性,通过kvc获取每个属性的值,并判断相应属性是否存在值转换器,若果存在,则对值进行相应转换。最后将所有的值存储在字典中,并调用modelWithDictionary: error通过kvc的方式验证并设置值。
方法九:modelWithDictionary: error
这个方法只是调用了initWithDictionary: error:自定义实现,而我在这个方法只是简单的调用了MTLModel的initWithDictionary: error:实现。
方法十:initWithDictionary: error:
这个方法通过kvc的方式初始化model实例。先是遍历了字典,获取值并验证是否为NSNull.null类型,接着通过KVC的方式为model实例验证并设置值。注意:XTrace跟踪不到静态函数。所以这里申明的MTLValidateAndSetValue函数并未跟踪到。
附上案例地址:GitHub - qsw1214/MantleDemo: A MantleDemo for learing
如果理解有错误的地方,欢迎大家可以指出来,一起探讨。