开发中,有时需要去打印模型中所有的属性的值,这时候需要我们去重写description
方法,并在description
方法中拼接所有属性的值。而每次在模型中重写description
方法是非常耽误开发速度并且影响心情。于是下面的解决方法就出现了。
在这个方法中主要运用一些runtime 的知识:
-
object_getClassName(id obj)
:得到实例对象
的类名
-
class_copyIvarList(Class cls, unsigned int *outCount)
:得到类的成员变量列表
。 -
ivar_getName(Ivar v)
:根据变量的到对应的变量名
-
ivar_getTypeEncoding(Ivar v)
: 得到变量对应的类型
基本思路:
- 在开发中,我们都会使用模型继承于一个
基模型
,我们就在基模型
做文章 - 利用在打印对象时,会调用
description
方法原理,重写基类模型
的description
方法。 - 调用子类没有实现的方法,会自动向上查询方法并调用方法。
- 这个方法可以下载NSObject分类中,也可以写在模型基类中
-(NSString *)description{
unsigned int count;
const char *clasName = object_getClassName(self);
NSMutableString *string = [NSMutableString stringWithFormat:@"<%s: %p>:[ \n",clasName, self];
Class clas = NSClassFromString([NSString stringWithCString:clasName encoding:NSUTF8StringEncoding]);
Ivar *ivars = class_copyIvarList(clas, &count);
for (int i = 0; i < count; i++) {
@autoreleasepool {
Ivar ivar = ivars[i];
const char *name = ivar_getName(ivar);
//得到类型
NSString *type = [NSString stringWithCString:ivar_getTypeEncoding(ivar) encoding:NSUTF8StringEncoding];
NSString *key = [NSString stringWithCString:name encoding:NSUTF8StringEncoding];
id value = [self valueForKey:key];
//确保BOOL 值输出的是YES 或 NO,这里的B是我打印属性类型得到的……
if ([type isEqualToString:@"B"]) {
value = (value == 0 ? @"NO" : @"YES");
}
[string appendFormat:@"\t%@: %@\n",[self delLine:key], value];
}
}
[string appendFormat:@"]"];
return string;
}
//因为ivar_getName得到的是一个带有下划线的名字,去掉下划线看起来更漂亮
-(NSString *)delLine:(NSString *)string{
if ([string hasPrefix:@"_"]) {
return [string substringFromIndex:1];
}
return string;
}
注意
- 记得
#import <objc/message.h>
- 在相互引用的类中,打印会出现死循环,目前没有找到解决方法,只能人为的去避免。
- 想到解决方法的小伙伴,记得留言,非常感谢。