天天看點

Object-C中nil, NULL跟NSNull

相信不少開發者,都被NSNull坑過,最常見的是伺服器傳回的json裡面,說好的字典、數組、數字,結果傳回的是空值。   這個時候,NSJSONSerialization 會自動把他們換成 NSNull。當我們再去用dict[@“hello”]的時候,就會出觸發exception,導緻程式崩潰。   那麼如何處理它呢? 我曾經的做法 寫了個宏,判斷傳回的這個類是不是NSNull類,即 isKindOfClass   最簡單的做法 相信大家都知道,[NSNull null] 并不是一個 工廠方法,而是一個 單例模式,那麼我們直接去判斷指派的這個指針是不是[NSNull null] 就好了。   那麼問題來了,編譯器會多了一個warning,很煩人。   在 這篇文章裡面介紹了各種做法:

  1. - (void)someMethod 
  2.     NSString *aString = @"loremipsum"; 
  3.     // This will complain: "Comparison of distinct pointer types ('NSString *' and 'NSNull *')" 
  4.     if (aString != [NSNull null]) 
  5.     { 
  6.     } 
  7.     // This works (at least for strings), but isEqual: does different things  
  8.     // for different classes, so it's not ideal 
  9.     if ([aString isEqual:[NSNull null]]) 
  10.     { 
  11.     } 
  12.     // If you cast it to the class you're comparing against 
  13.     // then you're good to go 
  14.     if (aString != (NSString *)[NSNull null]) 
  15.     { 
  16.     } 
  17.     // But we can also just cast it to id and 
  18.     // that works generically 
  19.     if (aString != (id)[NSNull null]) 
  20.     { 
  21.     } 
  22.     // The thing that would be really cool, 
  23.     // would be [NSNull null] returning 
  24.     // id (like in the sample category below). 
  25.     // Wouldn't count on that one though. 
  26.     if (aString != [NSNull idNull]) 
  27.     { 
  28.     } 

  這些都不是非常漂亮的解決方案,這篇文章的作者推薦:

  1. @interface NSNull (idNull) 
  2. + (id)idNull; 
  3. @end 
  4. @implementation NSNull (idNull) 
  5. + (id)idNull { return [NSNull null]; } 
  6. @end 

  或者呢

  1. if ([[NSNull null] isEqual:aString]) 

  最終解決方案 上面的做法,都需要判斷一次,還是很不優雅,為什麼呢,我們還是不能像NULL,nil一樣,直接拿來用,還是需要判斷一下,這裡推薦一套最漂亮的作法。   陳航提供了一個 gist   我發現我的octopress的gist插件挂了,直接貼出來好了。

  1. #define NSNullObjects @[@"",@0,@{},@[]] 
  2. @interface NSNull (InternalNullExtention) 
  3. @end 
  4. @implementation NSNull (InternalNullExtention) 
  5. - (NSMethodSignature*)methodSignatureForSelector:(SEL)selector 
  6.     NSMethodSignature* signature = [super methodSignatureForSelector:selector]; 
  7.     if (!signature) { 
  8.         for (NSObject *object in NSNullObjects) { 
  9.             signature = [object methodSignatureForSelector:selector]; 
  10.             if (signature) { 
  11.                 break; 
  12.             } 
  13.         } 
  14.     } 
  15.     return signature; 
  16. - (void)forwardInvocation:(NSInvocation *)anInvocation 
  17.     SEL aSelector = [anInvocation selector]; 
  18.     for (NSObject *object in NSNullObjects) { 
  19.         if ([object respondsToSelector:aSelector]) { 
  20.             [anInvocation invokeWithTarget:object]; 
  21.             return; 
  22.         } 
  23.     } 
  24.     [self doesNotRecognizeSelector:aSelector]; 
  25. @end 

  很高端霸氣上檔次的做法,通過處理異常情況,來實作這個功能。   這裡還提供一個 日本人的封裝方案:

  1. #import "NSNull+OVNatural.h" 
  2. @implementation NSNull (OVNatural) 
  3. - (void)forwardInvocation:(NSInvocation *)invocation 
  4.     if ([self respondsToSelector:[invocation selector]]) { 
  5.         [invocation invokeWithTarget:self]; 
  6.     } 
  7. - (NSMethodSignature *)methodSignatureForSelector:(SEL)selector 
  8.     NSMethodSignature *sig = [[NSNull class] instanceMethodSignatureForSelector:selector]; 
  9.     if(sig == nil) { 
  10.         sig = [NSMethodSignature signatureWithObjCTypes:"@^v^c"]; 
  11.     } 
  12.     return sig; 
  13. @end 

  關于[NSMethodSignature signatureWithObjCTypes:“@^vc”]的功能,可以參考以下兩篇文章: https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtTypeEncodings.html http://nshipster.com/type-encodings/

//判斷對象不空
if(object) {}

//判斷對象為空
if(object == nil) {}

//數組初始化,空值結束
NSArray *pageNames=[[NSArray alloc] initWithObjects:@"DocumentList",@"AdvancedSearch",@"Statistics",nil];

//判斷數組元素是否為空
UIViewController *controller=[NSArray objectAtIndex:i];
if((NSNull *)controller == [NSNull null])
{
    //
}

//判斷字典對象的元素是否為空
NSString *userId=[NSDictionary objectForKey:@"UserId"];
if(userId == [NSNull null])
{
    //
}