YYMode 源码学习 1

**YYMode 只有 5 个文件就实现了字典转模型,这是相当的牛逼! **

YYModel.h
NSObject+YYModel.h
NSObject+YYModel.m
YYClassInfo.h
YYClassInfo.m

YYMode 中的类:
NSObject+YYModel : 主要定义一些字典转模型,模型转字典的方法
NSArray+YYModel : 字典数组转模型数组
NSDictionary+YYModel : 字典转模型字典
protocol YYModel : 白名单,黑名单,模型属性和字典不匹配的转换,容器属性的映射等。

_YYModelMeta : 模型元
_YYModelPropertyMeta:模型属性元

YYClassInfo : 存储类的一些信息
YYClassIvarInfo : 存储变量的一些信息
YYClassMethodInfo : 存储方法的一些信息
YYClassPropertyInfo : 存储属性的一些信息

YYMode 特性:

  • 高性能: 转换性能接近手写代码。
  • 自动类型转换:可以自动转换的对象类型。
  • 类型安全:所有数据类型将验证以确保类型安全的转换过程。
  • 非侵入性:没有必要使模型类从其他基类继承。
  • 重量轻: 这个库仅包含5个文件。
  • 文档和单元测试覆盖:文档覆盖率100%, 代码覆盖率99.6%。

github 地址: https://github.com/ibireme/YYModel

字典转模型框架实现的基本功能:

  1. 字典转模型
    YYMode要实现的转换过程:
    json 字符串 ——> data ——> dict ——> mode
    最需要 YYMode 做的也就是 dict ——> mode 这一步。
  2. 模型转字典
    mode ——> dict ——> data ——>json字符串
    最需要 YYMode 做的也就是 mode ——> dict 这一步。

你可能需要问:json 字符串 ——> data ——> dictdict ——> data ——>json 字符串 这几步的转换谁给我们做了。

答案: 是 apple 给我做了。

json 字符串 ——> data ——> dict

// 定义一个 json 字符串
NSString *jsonStr = @"{\"name\" : \"json\"}";

// 字符串转 data 
NSData *data = [jsonStr dataUsingEncoding:NSUTF8StringEncoding];

// data 转 dict 或者 array
NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil];


// 打印结果
{
    name = json;
}

dict ——> data ——>json

// 创建一个 dict 
NSDictionary *dict  =  @{@"name" : @"json"};

// 将 dict 转换为 data
NSData *newData = [NSJSONSerialization dataWithJSONObject:dict options:0 error:nil];

// data 转字符串
NSString *newJsonStr = [[NSString alloc] initWithData:newData encoding:NSUTF8StringEncoding];


// 打印结果 
{"name":"json"}     // 不要和我说,我上例定义的字符串和打印的字符串不一样。

这几步就这么愉快的搞定了。(YYMode 具体怎么做的还要看源码)

看源码最快的方式是找主要功能方法追根溯源!
我们现在就根据 YYMode 的主要功能代码,查看字典转模型的实现原理。

作者给出的实例代码

// JSON:
{
    "uid":123456,
    "name":"Harry",
    "created":"1965-07-31T00:00:00+0000"
}

// Model:
@interface User : NSObject
@property UInt64 uid;
@property NSString *name;
@property NSDate *created;
@end
@implementation User
@end


// Convert json to model:
User *user = [User yy_modelWithJSON:json];

这个方法就是主要的字典转模型的方法:
+ (nullable instancetype)yy_modelWithJSON:(id)json;

方法的具体实现:


+ (instancetype)yy_modelWithJSON:(id)json {

    // 解析 json 对象 为 字典对象
    NSDictionary *dic = [self _yy_dictionaryWithJSON:json];

    // 字典转模型操作
    return [self yy_modelWithDictionary:dic];
}

解析 json 对象 为 字典对象

+ (NSDictionary *)_yy_dictionaryWithJSON:(id)json {

    // 判断 json 对象是否为空,为空就直接返回
    if (!json || json == (id)kCFNull) return nil;


    NSDictionary *dic = nil;
    NSData *jsonData = nil;

    // 传入的 json 为 dict 就 直接记录
    if ([json isKindOfClass:[NSDictionary class]]) {
        dic = json;

    // 传入的 json 为 string 就 转为 data 
    } else if ([json isKindOfClass:[NSString class]]) {
        jsonData = [(NSString *)json dataUsingEncoding : NSUTF8StringEncoding];

    // 传入的 json 为 data 就 直接记录
    } else if ([json isKindOfClass:[NSData class]]) {
        jsonData = json;
    }

    // 这里是进行 data 转字典的操作
    if (jsonData) {
        dic = [NSJSONSerialization JSONObjectWithData:jsonData options:kNilOptions error:NULL];
        if (![dic isKindOfClass:[NSDictionary class]]) dic = nil;
    }
    return dic;
}

处理的整体过程和我的 json 字符串 ——> data ——> dict 一致。

** 字典转模型 **

+ (instancetype)yy_modelWithDictionary:(NSDictionary *)dictionary {

    // 判断传入的字典是否是空 ,为空直接返回
    if (!dictionary || dictionary == (id)kCFNull) return nil;

    // 传入的数据不是字典数据直接返回
    if (![dictionary isKindOfClass:[NSDictionary class]]) return nil;
    

    // 主要是对类的信息进行一些缓存处理
     /*
        1. 类的白名单
        2. 类的黑名单
        3. 容器属性类的处理
        4. 类的属性列表
        5. 类的变量列表
        6. 类的方法列表
          等等!
    */
    Class cls = [self class];
    _YYModelMeta *modelMeta = [_YYModelMeta metaWithClass:cls];
    if (modelMeta->_hasCustomClassFromDictionary) {
        cls = [cls modelCustomClassForDictionary:dictionary] ?: cls;
    }
    
    // 通过自己的 “类” 创建一个模型对象
    NSObject *one = [cls new];

    // 进行字典转模型操作 (转换成功直接返回,转换失败返回 nil )
    // 由于对类的信息已经进行了处理,这里其实就是简单的赋值操作 
    if ([one yy_modelSetWithDictionary:dictionary]) return one;
    return nil;
}
- (BOOL)yy_modelSetWithDictionary:(NSDictionary *)dic {

    // 传入的字典为 nil 直接字典转模型失败
    if (!dic || dic == (id)kCFNull) return NO;
    
    // dict 不是字典直接转换失败
    if (![dic isKindOfClass:[NSDictionary class]]) return NO;
    
    // 获取 元类 模型缓存
    _YYModelMeta *modelMeta = [_YYModelMeta metaWithClass:object_getClass(self)];
      
    // 元类 模型的 映射 key 和 keyPath 的个数是 0  , 直接转换失败
    if (modelMeta->_keyMappedCount == 0) return NO;
    
    // 自定义转换
    if (modelMeta->_hasCustomWillTransformFromDictionary) {
        dic = [((id<YYModel>)self) modelCustomWillTransformFromDictionary:dic];
        if (![dic isKindOfClass:[NSDictionary class]]) return NO;
    }
    
    // 模型设置上下文 (暂时没明白是干嘛的)
    ModelSetContext context = {0};
    
    // 元模型
    context.modelMeta = (__bridge void *)(modelMeta);
    // 模型
    context.model = (__bridge void *)(self);
    // 字典
    context.dictionary = (__bridge void *)(dic);
    
    // 模型中的 key value 的映射基本是在这完成的 
    // 模型的 映射 key 和 keyPath 大于等于  字典中元素的个数
    if (modelMeta->_keyMappedCount >= CFDictionaryGetCount((CFDictionaryRef)dic)) {

        // 
        CFDictionaryApplyFunction((CFDictionaryRef)dic, ModelSetWithDictionaryFunction, &context);

        // 路径属性
        if (modelMeta->_keyPathPropertyMetas) {
            CFArrayApplyFunction((CFArrayRef)modelMeta->_keyPathPropertyMetas,
                                 CFRangeMake(0, CFArrayGetCount((CFArrayRef)modelMeta->_keyPathPropertyMetas)),
                                 ModelSetWithPropertyMetaArrayFunction,
                                 &context);
        }

        // 多路径属性
        if (modelMeta->_multiKeysPropertyMetas) {
            CFArrayApplyFunction((CFArrayRef)modelMeta->_multiKeysPropertyMetas,
                                 CFRangeMake(0, CFArrayGetCount((CFArrayRef)modelMeta->_multiKeysPropertyMetas)),
                                 ModelSetWithPropertyMetaArrayFunction,
                                 &context);
        }
    } else {
        CFArrayApplyFunction((CFArrayRef)modelMeta->_allPropertyMetas,
                             CFRangeMake(0, modelMeta->_keyMappedCount),
                             ModelSetWithPropertyMetaArrayFunction,
                             &context);
    }
    
    if (modelMeta->_hasCustomTransformFromDictionary) {
        return [((id<YYModel>)self) modelCustomTransformFromDictionary:dic];
    }
    return YES;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容