Objective-C 中 NULL、nil、Nil、NSNull 的定义及不同

在C语言中用0来作为"不存在"的原始值,而用NULL作为指针空值。在Objective-C中,则有几种不同的方式来表示"不存在",分别有:NULL、nil、Nil、NSNull。下面我们来看看这几种空值的定义以及使用上的不同。
注: 以下各种空值定义的源码摘自iOS 10.0 SDK中的相关头文件

NULL

NULL定义在user/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++环境下不同编译器的_ _DARWIN_NULL的取值,然后定了其他环境下_ _DARWIN_NULL的值,因此在Objective-C中NULL的最终定义为:

#define NULL ((void*)0)

即NULL本质上是:(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)用于判断当前环境是否有C++的nullptr特性,如果有.nil定义为nullptr,否则nil定义为**_ _DARWIN_NULL,所以在Objective-C中nil的最终定义为:

#define nil ((void*)0)

也就是说,nil本质上也是:(void *)0,与NULL一致。

使用惯例:nil用于表示指向Objective-C对象(id类型的对象,或者使用@interface声明的OC对象)的指针为空,例如:

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本质上也是:(void *)0。

使用惯例:Nil用于表示指向Objective-C类(Class)类型的指针为空,例如:

Class someClass = Nil;
Class anotherClass = [NSString class];

NSNull

NSNull定义在NSNull.h文件里:

#import 
 
NS_ASSUME_NONNULL_BEGIN
 
@interface NSNull : NSObject 
 
+ (NSNull *)null;
 
@end
 
NS_ASSUME_NONNULL_END

从上述定义中,我们可知NSNull是一个Objective-C对象,是一个用于表示空值得类,而且它只有一个单例方法:+[NSNull null],一般用于在集合对象中保存一个空的占位对象。

使用惯例:在Foundation集合对象(NSArray、NSDictionary、NSSet等)中,nil通常被用于表示集合对象结束的标志,因此无法用nil来存储一个空值,所以一般用[NSNull null]空对象来存储。另外,在NSDictionary的-objectForKey:方法中,如果当前字典中key对应的值不存在,该方法会返回nil,表明当前key在字典中未添加,但是如果我们想明确表示某一key已经在字典中添加,但是它没有值,这时候就可以用[NSNull null]来赋值表示。

// 当 NSArray 里遇到 nil 时,就说明这个数组对象的元素截止了,即 NSArray 只关注 nil 之前的对象,nil 之后的对象会被抛弃。
NSArray *array = [NSArray arrayWithObjects:@"one", @"two", nil];
 
// 错误的使用
NSMutableDictionary *dict = [NSMutableDictionary dictionary];
[dict setObject:nil forKey:@"someKey"];
 
// 正确的使用
NSMutableDictionary *dict = [NSMutableDictionary dictionary];
[dict setObject:[NSNull null] forKey:@"someKey"];

NIL或NSNil

Objective-C中不存在这两个符号!!!

总结

从上述分析我们可知,不管是NULL、nil还是Nil,它们本质上是一样的,都是(void )0,只是写法不同。这样做的意义是为了区分不同的数据类型,虽然它们值相同,但我们需要理解它们之间的字面意义并用于不同场景,让代码更加准确,增加可读性。
标志 值 含义
NULL (void
)0 C指针的字面空值
nil (id)0 Objective-C 对象的字面空值
Nil (Class)0 Objective-C 类的字面空值
NSNull [NSNull null] 用来表示空值的Objective-C对象

Objective-C 中 NULL、nil、Nil、NSNull 的定义及不同_第1张图片

你可能感兴趣的:(Objective-C 中 NULL、nil、Nil、NSNull 的定义及不同)