输出和debug在老师儿看来一样有很大学问,下面来谈一下:数组
调试程序时,常常须要打印并查看对象信息。一种比较傻比的方法是把对象的全部属性一一输出到日志中。可是最经常使用的仍是这种直接输出。atom
NSLog(@"arr = %@",array);spa
在构建须要打印到日志的字符串时,array对象会收到description消息,该方法所返回的描述信息将取代“各式字符串”(format string)里的“%@”。比方说,object是个数组,若用下列代码打印其信息:.net
NSArray *array = @[@"lao shi",@"didi",@"hou",@(222)];debug
NSLog(@"arr = %@",array);指针
则会输出:调试
arr = (日志
"lao shi",code
didi,orm
hou,
222
)
然而若是输出自定义的类对象时,输出的是
<CustomAlertView: 0x14643e30>
与array为数组时所输出的信息相比,上面这种内容不太有用。除非在本身的类里覆写dexcription方法,不然打印信息时就会调用NSObject类所实现的默认方法。此方法定义在NSObject协议里,不过NSObject类也实现它。由于NSObject并非惟一的“根类”,因此许多方法都要定义再NSObject协议里。比方说NSProxy也是一个听从了NSObject协议的“根类”。因为description等方法定义在NSObject协议里,所以像NSProxy这种“根类”及其子类也必须实现他们。如前所见,这些实现好的方法并无打印出较为有用的内容,只不过是输出了类名和对像的内存地址。只有在你想判断两指针是否真的指向同一对象时,这种信息才有用处。除此以外,再也看不出其余有用的内容了。咱们想打印出来的对象信息应该比这更多才对。
要向输出更为有用的信息也很简单,只需覆写description方法并将描述此对象的字符串返回便可。例如,有下面这个表明我的信息的类:
@interface LaoShiEr : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, copy) NSString *age;
- (id)initWithName:(NSString *)name age:(NSString *)age;
@implementation LaoShiEr
- (id)initWithName:(NSString *)name age:(NSString *)age
{
if ((self = [super init])) {
_name = name;
_age = age;
}
return self;
}
该类的description方法一般能够这样实现:
- (NSString *)description
{
return [NSString stringWithFormat:@"<%@: %p,\"%@ %@\">",[self class],self,_name,_age];
}
假如按上面的代码来写,那么LaoShiEr对象就会输出以下格式的信息:
@interface LaoShiEr : NSObject
@property (nonatomic, copy, readonly) NSString *name;
@property (nonatomic, copy, readonly) NSString *work;
- (id)initWithName:(NSString *)name
work:(NSString *)work;
@implementation LaoShiEr
- (id)initWithName:(NSString *)name
work:(NSString *)work
{
if ((self = [super init])) {
_name = [name copy];
_work = [work copy];
}
return self;
}
- (NSString *)description
{
return [NSString stringWithFormat:@"<%@ : %p,\"%@ %@\">",[self class],self,_name,_work];
}
按照上面的代码来写,输出以下信息
LaoShiEr *laoshi = [[LaoShiEr alloc]
initWithName:@"laoshier"
work:@"coder"];
NSLog(@"laoshier = %@",laoshi);
laoshier = <LaoShiEr : 0xb64bec0,"laoshier coder">
这样就比以前所输出得信息更加清楚了,也更为有用了。再description中输出不少互不相同的信息的时候能够借助NSDictionary类的description方法。修改一下老师儿的description方法
- (NSString *)description
{
return [NSString stringWithFormat:@"<%@ : %p, %@>",[self class],self,@{@"name":_name,@"work":_work}];
}
再此输出
laoshier = <LaoShiEr : 0xb677420, {
name = laoshier;
work = coder;
}>
这样是否是看上去很是帅气啊NSObject协议中还有个方法要注意,那就是debugDescription,此方法的用意与description很是想似。两者区别在于,debugDescription方法是开发者在调试器中以控制台命令打印对象时才调用的。在NSObject类的默认实现中,此方法只是直接调用了description。以LaoShiEr为例,咱们在建立实例所用的代码后面插入断点,而后经过调试器运行程序,暂停于此:而且po完成对象打印:
<LaoShiEr : 0xb7c4310, {
name = laoshier;
work = coder;
}>
当你不想把类名与指针地址这种额外内容放在普通的描述信息里,可是却但愿调试的时候可以很方便地看到它们,在此状况下,就可使用这种输出方式来实现。- (NSString *)description
{
return [NSString stringWithFormat:@"<%@>",@{@"name":_name,@"work":_work}];
}
- (NSString *)debugDescription
{
return [NSString stringWithFormat:@"<%@ : %p, %@>",[self class],self,@{@"name":_name,@"work":_work}];
}
好的,今天就到这里,真正用的到位的话,调试起来会很方便,很省时间,提升效率的。