本文由我們團(tuán)隊(duì)的 康祖彬 童鞋撰寫,這是他的個(gè)人主頁:https://kangzubin.cn峡扩。
理解”不存在“的概念不僅僅是一個(gè)哲學(xué)的問題蹭越,也是一個(gè)實(shí)際的問題。我們是有形宇宙的居民教届,而原因在于邏輯宇宙的存在不確定性响鹃。作為一個(gè)邏輯系統(tǒng)的物理體現(xiàn)驾霜,計(jì)算機(jī)面臨一個(gè)棘手的問題,就是如何用”存在“表達(dá)”不存在“买置。--摘自 NSHipster
這段話讀起來怪怪的粪糙,畢竟是翻譯過來的,大概意思是說在計(jì)算機(jī)中如何描述”不存在“這個(gè)概念很重要忿项。
在 C 語言中用 0
來作為“不存在”的原始值蓉冈,而用 NULL
作為指針空值。在 Objective-C 中倦卖,則有幾種不同的方式來表示“不存在”洒擦,分別有:NULL
、nil
怕膛、Nil
熟嫩、NSNull
。下面我們來看看這幾種空值的定義以及使用上的不同褐捻。
注:以下各種空值定義的源碼摘自 iOS 10.0 SDK 中的相關(guān)頭文件掸茅。
NULL
NULL
定義在 usr/include/sys/_types/_null.h
文件里:
#ifndef NULL
#define NULL __DARWIN_NULL
#endif /* NULL */
其中 __DARWIN_NULL
的定義在 usr/include/sys/__types.h
文件里,如下:
#ifdef __cplusplus
# ifdef __GNUG__
# define __DARWIN_NULL __null
# else /* ! __GNUG__ */
# ifdef __LP64__
# define __DARWIN_NULL (0L)
# else /* !__LP64__ */
# define __DARWIN_NULL 0
# endif /* __LP64__ */
# endif /* __GNUG__ */
#else /* ! __cplusplus */
# define __DARWIN_NULL ((void *)0)
#endif /* __cplusplus */
上述代碼首先定義在 C++ 環(huán)境下不同編譯器的 __DARWIN_NULL
的取值柠逞,然后定了其他環(huán)境下 __DARWIN_NULL
的值昧狮,因此在 Objective-C 中 NULL
的最終定義為:
#define NULL ((void*)0)
即 NULL
本質(zhì)上是:(void*)0
。
使用慣例:NULL
一般用于表示 C 指針空值板壮,例如:
int *pointerToInt = NULL;
char *pointerToChar = NULL;
struct TreeNode *rootNode = NULL;
nil
nil
定義在 usr/include/objc/objc.h
文件里:
#ifndef nil
# if __has_feature(cxx_nullptr)
# define nil nullptr
# else
# define nil __DARWIN_NULL
# endif
#endif
其中 __has_feature(cxx_nullptr)
用于判斷當(dāng)前環(huán)境是否有 C++ 的 nullptr 特性逗鸣,如果有,nil
定義為 nullptr
绰精,否則 nil
定義為 __DARWIN_NULL
撒璧,所以在 Objective-C 中 nil
的最終定義為:
#define nil ((void*)0)
也就是說,nil
本質(zhì)上也是:(void *)0
笨使,與 NULL
一致卿樱。
使用慣例:nil
用于表示指向 Objective-C 對(duì)象(id 類型的對(duì)象,或者使用 @interface 聲明的 OC 對(duì)象)的指針為空硫椰,例如:
NSString *someString = nil;
NSURL *someURL = nil;
id someObject = nil;
if (anotherObject == nil) // do something
Nil
Nil
定義在 usr/include/objc/objc.h
文件里:
#ifndef Nil
# if __has_feature(cxx_nullptr)
# define Nil nullptr
# else
# define Nil __DARWIN_NULL
# endif
#endif
與上述 nil
一致繁调,Nil
本質(zhì)上也是:(void *)0
。
使用慣例:Nil
用于表示指向 Objective-C 類(Class)類型的指針為空靶草,例如:
Class someClass = Nil;
Class anotherClass = [NSString class];
NSNull
NSNull
定義在 NSNull.h
文件里:
#import <Foundation/NSObject.h>
NS_ASSUME_NONNULL_BEGIN
@interface NSNull : NSObject <NSCopying, NSSecureCoding>
+ (NSNull *)null;
@end
NS_ASSUME_NONNULL_END
從上述定義中蹄胰,我們可知 NSNull
是一個(gè) Objective-C 對(duì)象,是一個(gè)用于表示空值的類奕翔,而且它只有一個(gè)單例方法:+[NSNull null]烤送,一般用于在集合對(duì)象中保存一個(gè)空的占位對(duì)象。
使用慣例:在 Foundation 集合對(duì)象(NSArray糠悯、NSDictionary帮坚、NSSet 等)中, nil
通常被用于表示集合對(duì)象結(jié)束的標(biāo)志互艾,因此無法用 nil
來存儲(chǔ)一個(gè)空值试和,所以一般用 [NSNull null]
空對(duì)象來存儲(chǔ)。另外纫普,在 NSDictionary 的 -objectForKey:
方法中阅悍,如果當(dāng)前字典中 key 對(duì)應(yīng)的值不存在時(shí),該方法會(huì)返回 nil
昨稼,表明當(dāng)前 key 在字典中未添加节视,但是如果我們想明確表示某一 key 已經(jīng)在字典中添加,但是它沒有值假栓,這時(shí)候就可以用 [NSNull null]
來賦值表示寻行。
// 當(dāng) NSArray 里遇到 nil 時(shí),就說明這個(gè)數(shù)組對(duì)象的元素截止了匾荆,即 NSArray 只關(guān)注 nil 之前的對(duì)象拌蜘,nil 之后的對(duì)象會(huì)被拋棄。
NSArray *array = [NSArray arrayWithObjects:@"one", @"two", nil];
// 錯(cuò)誤的使用
NSMutableDictionary *dict = [NSMutableDictionary dictionary];
[dict setObject:nil forKey:@"someKey"];
// 正確的使用
NSMutableDictionary *dict = [NSMutableDictionary dictionary];
[dict setObject:[NSNull null] forKey:@"someKey"];
NIL 或 NSNil
Objective-C 中不存在這兩個(gè)符號(hào)Q览觥<蛭浴!
總結(jié)
從上述分析我們可知烤芦,不管是 NULL
举娩、nil
還是 Nil
,它們本質(zhì)上是一樣的构罗,都是 (void *)0
铜涉,只是寫法不同。這樣做的意義是為了區(qū)分不同的數(shù)據(jù)類型绰播,雖然它們值相同骄噪,但我們需要理解它們之間的字面意義并用于不同場景,讓代碼更加明確蠢箩,增加可讀性链蕊。
標(biāo)志 | 值 | 含義 |
---|---|---|
NULL | (void *)0 | C 指針的字面空值 |
nil | (id)0 | Objective-C 對(duì)象的字面空值 |
Nil | (Class)0 | Objective-C 類的字面空值 |
NSNull | [NSNull null] | 用來表示空值的 Objective-C 對(duì)象 |