iOS Kingdom — 模型信息输出

Mouse
在 iOS 开发中,相信有很多程序员都会遇到过这种事情 —— 当你用 NSLog 打印一个 NSArrayNSDictionary 时,当数组或字典中有自定义类时,控制台只会输出这个类的类名加上其内存地址,如下图:
模型名称和内存地址
如果我们要看其中某个元素的某个属性值,则需要将这个元素取出来,然后再输出对应的元素值,大致代码如下:
// model0 model1 是 WQModel 的两个实例
NSArray *arr = @[model0, model1];
WQModel *model = arr[0];
NSLog(@"name = %@",model.name);
然而这几句代码在实际中的作用是查看 modelname 的值,对你的业务逻辑没有什么实质性的贡献。如果我们能在打印数组元素时就可以看到其元素中属性的值则可以少写一些代码,又可以更直观的看出其内容,如下图:
模型内容输出
要实现 图:模型内容输出 的日志打印形式,首先须要清楚这几个函数:
-description :当你输出一个对象时会调用该函数,如:NSLog(@"%@",model);
-debugDescription :当你在使用 LLDB 在控制台输入 po model 时会调用该函数
-descriptionWithLocale:indent: :存在于 NSArrayNSDictionary 等类中。当类中有这个函数时,它的优先级为 -descriptionWithLocale:indent: > -description

1、NSObject+WQModelNSObjectDescription 函数实现

重写 -description-debugDescription,返回对象的输出描述
- (NSString *)description {
    return [self modelDescriptionWithIndent:0];
}

- (NSString *)debugDescription {
    return [self modelDescriptionWithIndent:0];
}

- (NSString *)modelDescriptionWithIndent:(NSInteger)level {
    uint count;
    objc_property_t *properties = class_copyPropertyList([self class], &count);
    NSMutableString *mStr = [NSMutableString string];
    NSMutableString *tab = [NSMutableString stringWithString:@""];
    for (int index = 0; index < level; index ++) {
        [tab appendString:@"\t"];
    }
    [mStr appendString:@"{\n"];
    for (int index = 0; index < count; index ++) {
        NSString *lastSymbol = index + 1 == count ? @"" : @";";
        objc_property_t property = properties[index];
        NSString *name = @(property_getName(property));
        id value = [self valueForKey:name];
        [self dictionaryFormatWithMStr:mStr
                                   tab:tab
                                  name:name
                                 value:value
                            lastSymbol:lastSymbol
                                indent:level];
    }
    [mStr appendFormat:@"%@}",tab];
    free(properties);
    return [NSString stringWithFormat:@"<%@ : %p> %@",
            NSStringFromClass([self class]),
            self,
            mStr];
}

// 定义字典输出格式
- (void)dictionaryFormatWithMStr:(NSMutableString *)mStr
                             tab:(NSMutableString *)tab
                            name:(id)name
                           value:(id)value
                      lastSymbol:(NSString *)lastSymbol
                          indent:(NSInteger)level{
    if ([NSBundle bundleForClass:[value class]] == [NSBundle mainBundle]) {
        // 自定义类
        if ([value respondsToSelector:@selector(modelDescriptionWithIndent:)]) {
            [mStr appendFormat:@"\t%@%@ = %@%@\n",
             tab,
             name,
             [value modelDescriptionWithIndent:level + 1],
             lastSymbol];
            return;
        }
    }else {
        // 系统类
        if ([value respondsToSelector:@selector(descriptionWithLocale:indent:)]) {
            [mStr appendFormat:@"\t%@%@ = %@%@\n",
             tab,
             name,
             [value descriptionWithLocale:[NSLocale systemLocale]
                                   indent:level + 1],
             lastSymbol];
            return;
        }
    }
    [mStr appendFormat:@"\t%@%@ = %@%@\n",
     tab,
     name,
     value,
     lastSymbol];
}

// 定义数组输出格式
- (void)arrayFormatWithMStr:(NSMutableString *)mStr
                        tab:(NSMutableString *)tab
                      value:(id)value
                 lastSymbol:(NSString *)lastSymbol
                     indent:(NSInteger)level {
    if ([NSBundle bundleForClass:[value class]] == [NSBundle mainBundle]) {
        // 自定义类
        if ([value respondsToSelector:@selector(modelDescriptionWithIndent:)]) {
            [mStr appendFormat:@"\t%@%@%@\n",
             tab,
             [value modelDescriptionWithIndent:level + 1],
             lastSymbol];
            return;
        }
    }else {
        // 系统类
        if ([value respondsToSelector:@selector(descriptionWithLocale:indent:)]) {
            [mStr appendFormat:@"\t%@%@%@\n",
             tab,
             [value descriptionWithLocale:[NSLocale systemLocale]
                                   indent:level + 1],
             lastSymbol];
            return;
        }
    }
    [mStr appendFormat:@"\t%@%@%@\n",
     tab,
     value,
     lastSymbol];
}

2、NSObject+WQModelNSArrayDescription 函数实现

重写 -descriptionWithLocale:indent:-debugDescription,返回对象的输出描述
- (NSString *)descriptionWithLocale:(id)locale
                             indent:(NSUInteger)level {
    NSMutableString *mStr = [NSMutableString string];
    NSMutableString *tab = [NSMutableString stringWithString:@""];
    for (int index = 0; index < level; index ++) {
        [tab appendString:@"\t"];
    }
    [mStr appendString:@"(\n"];
    for (int index = 0; index < self.count; index ++) {
        NSString *lastSymbol = (self.count == index + 1) ? @"":@",";
        id value = self[index];
        [self arrayFormatWithMStr:mStr
                              tab:tab
                            value:value
                       lastSymbol:lastSymbol
                           indent:level];
    }
    [mStr appendFormat:@"%@)",tab];
    return mStr;
}

- (NSString *)debugDescription {
    return [self descriptionWithLocale:[NSLocale systemLocale]];
}

3、NSObject+WQModelNSDictionaryDescription 函数实现

重写 -descriptionWithLocale:indent:-debugDescription,返回对象的输出描述
- (NSString *)descriptionWithLocale:(id)locale
                             indent:(NSUInteger)level {
    NSMutableString *mStr = [NSMutableString string];
    NSMutableString *tab = [NSMutableString stringWithString:@""];
    for (int index = 0; index < level; index++) {
        [tab appendString:@"\t"];
    }
    [mStr appendString:@"{\n"];
    NSArray *allKey = self.allKeys;
    for (int index = 0; index < allKey.count; index++) {
        id value = self[allKey[index]];
        NSString *lastSymbol = (allKey.count == index + 1) ? @"":@";";
        [self dictionaryFormatWithMStr:mStr
                                   tab:tab
                                  name:allKey[index]
                                 value:value
                            lastSymbol:lastSymbol
                                indent:level];
    }
    [mStr appendFormat:@"%@}",tab];
    return mStr;
}

- (NSString *)debugDescription {
    return [self descriptionWithLocale:[NSLocale systemLocale]];
}

4、NSObject+WQModelNSSetDescription 函数实现

重写 -descriptionWithLocale:indent:-debugDescription,返回对象的输出描述
- (NSString *)descriptionWithLocale:(id)locale
                             indent:(NSUInteger)level {
    NSMutableString *mStr = [NSMutableString string];
    NSMutableString *tab = [NSMutableString stringWithString:@""];
    for (int index = 0; index <level; index ++) {
        [tab appendString:@"\t"];
    }
    [mStr appendString:@"{(\n"];
    for (id value in [self allObjects]) {
        NSString *lastSymbol = [value isEqual:[[self allObjects] lastObject]] ? @"" : @",";
        [self arrayFormatWithMStr:mStr
                              tab:tab
                            value:value
                       lastSymbol:lastSymbol
                           indent:level];
    }
    [mStr appendFormat:@"%@)}",tab];
    return mStr;
}

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

推荐阅读更多精彩内容

  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,469评论 33 467
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,293评论 18 399
  • 古有琴师伯牙在钟子期死后,摔琴绝弦,终身不弹,故有高山流水之曲。而今,朋友之间却因意见相左而分道扬镳!既然圈子...
    清泠若馨阅读 524评论 2 1
  • 从小到大听妈妈说的最多的话,便是在家靠父母,出门靠朋友。的确我也对这句话深信不疑,所以即便是在学校我也要努力维持和...
    谭谭ne阅读 233评论 0 1