美文网首页
重写Model的description 属性

重写Model的description 属性

作者: 葬己 | 来源:发表于2017-07-06 11:26 被阅读49次

    调试程序时,经常需要打印并查看对象信息。一种比较傻比的方法是把对象的所有属性一一输出到日志中。但是最常用的还是这种直接输出。

     NSLog(@"arr = %@",array);
    

    在构建需要打印到日志的字符串时,array对象会收到description消息,该方法所返回的描述信息将取代“各式字符串”(format string)里的“%@”。比方说,object是个数组,若用下列代码打印其信息:

    NSArray *array =@[@"lao shi",@"didi",@"hou",@(222)];
    NSLog(@"arr = %@",array);
    

    则会输出:

    arr = (
             "lao shi",
            didi,
            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;
    @end
    
    
    @implementation LaoShiEr
    -  (id)initWithName:(NSString *)name age:(NSString *)age{
      if ((self = [superinit])) {
           _name = name;
           _age = age;
        }
        return self;
    }
    
    @end
    

    该类的description方法通常可以这样实现:

    - (NSString *)description{
       return [NSStringstringWithFormat:@"<%@: %p,\"%@ %@\">",[selfclass],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;
      @end
    
    @implementation LaoShiEr
    -  (id)initWithName:(NSString *)name work:(NSString *)work{
          if  ((self = [superinit])) {
               _name = [name copy];
               _work = [work copy];
         }
         return self;
     }
    
     -  (NSString *)description{
         return [NSStringstringWithFormat:@"<%@ : %p,\"%@ %@\">",[selfclass],self,_name,_work];
     }
    
     @end
    

    按照上面的代码来写,输出如下信息

    LaoShiEr *laoshi = [[LaoShiEralloc]
                        initWithName:@"laoshier"
                        work:@"coder"];
    
     NSLog(@"laoshier = %@",laoshi);
    
     laoshier = <LaoShiEr : 0xb64bec0,"laoshier coder">
    

    这样就比之前所输出得信息更加清楚了,也更为有用了。再description中输出很多互不相同的信息的时候可以借助NSDictionary类的description方法。修改一下老师儿的description方法

       -  (NSString *)description{
          return [NSStringstringWithFormat:@"<%@ : %p, %@>",[selfclass],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 [NSStringstringWithFormat:@"<%@>",@{@"name":_name,@"work":_work}];
    }

      - (NSString *)debugDescription{
         return [NSStringstringWithFormat:@"<%@ : %p, %@>",[selfclass],self,@{@"name":_name,@"work":_work}];
      }
    

    好的,今天就到这里,真正用的到位的话,调试起来会很方便,很省时间,提高效率的。

    相关文章

      网友评论

          本文标题:重写Model的description 属性

          本文链接:https://www.haomeiwen.com/subject/xtkhhxtx.html