Objective-C 中 NULL、nil颅和、Nil傅事、NSNull 的定義及不同

本文由我們團(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 中倦卖,則有幾種不同的方式來表示“不存在”洒擦,分別有:NULLnil怕膛、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ì)象

Reference

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末滔韵,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子掌实,更是在濱河造成了極大的恐慌陪蜻,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,635評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件贱鼻,死亡現(xiàn)場離奇詭異宴卖,居然都是意外死亡滋将,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,543評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門症昏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來随闽,“玉大人,你說我怎么就攤上這事肝谭【蛳埽” “怎么了?”我有些...
    開封第一講書人閱讀 168,083評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵攘烛,是天一觀的道長魏滚。 經(jīng)常有香客問我,道長坟漱,這世上最難降的妖魔是什么鼠次? 我笑而不...
    開封第一講書人閱讀 59,640評(píng)論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮靖秩,結(jié)果婚禮上须眷,老公的妹妹穿的比我還像新娘。我一直安慰自己沟突,他們只是感情好花颗,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,640評(píng)論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著惠拭,像睡著了一般扩劝。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上职辅,一...
    開封第一講書人閱讀 52,262評(píng)論 1 308
  • 那天棒呛,我揣著相機(jī)與錄音,去河邊找鬼域携。 笑死簇秒,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的秀鞭。 我是一名探鬼主播趋观,決...
    沈念sama閱讀 40,833評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼锋边!你這毒婦竟也來了皱坛?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,736評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤豆巨,失蹤者是張志新(化名)和其女友劉穎剩辟,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,280評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡贩猎,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,369評(píng)論 3 340
  • 正文 我和宋清朗相戀三年熊户,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片融欧。...
    茶點(diǎn)故事閱讀 40,503評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡敏弃,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出噪馏,到底是詐尸還是另有隱情,我是刑警寧澤绿饵,帶...
    沈念sama閱讀 36,185評(píng)論 5 350
  • 正文 年R本政府宣布欠肾,位于F島的核電站,受9級(jí)特大地震影響拟赊,放射性物質(zhì)發(fā)生泄漏刺桃。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,870評(píng)論 3 333
  • 文/蒙蒙 一吸祟、第九天 我趴在偏房一處隱蔽的房頂上張望瑟慈。 院中可真熱鬧,春花似錦屋匕、人聲如沸葛碧。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,340評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽进泼。三九已至,卻和暖如春纤虽,著一層夾襖步出監(jiān)牢的瞬間乳绕,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,460評(píng)論 1 272
  • 我被黑心中介騙來泰國打工逼纸, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留洋措,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,909評(píng)論 3 376
  • 正文 我出身青樓杰刽,卻偏偏與公主長得像菠发,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子专缠,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,512評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容