您的位置:首页 > 移动开发 > IOS开发

iOS开发-NULL和nullptr和nil和Nil还有NSNull

2016-07-13 00:00 507 查看
NULL和nullptr

在Clang6.0的stddef.h文件中可以找到NULL和nullptr的声明:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
#undefNULL
#ifdef__cplusplus
#if!defined(__MINGW32__)&&!defined(_MSC_VER)
#defineNULL__null
#else
#defineNULL0
#endif
#else
#defineNULL((void*)0)
#endif
#ifdef__cplusplus
#ifdefined(_MSC_EXTENSIONS)&&defined(_NATIVE_NULLPTR_SUPPORTED)
namespacestd{typedefdecltype(nullptr)nullptr_t;}
using::std::nullptr_t;
#endif
#endif
早在1972年,C语言诞生的初期,常数0带有常数及空指针的双重身分。C使用preprocessormacroNULL表示空指针,让NULL及0分别代表空指针及常数0。NULL可被定义为((void*)0)或是0。

C++并不采用C的规则,不允许将void*隐式转换为其他类型的指针。为了使代码char*c=NULL;能通过编译,NULL只能定义为0。这样的决定使得函数重载无法区分代码的语义:

1
2
voidfoo(char*);
voidfoo(int);
C++建议NULL应当定义为0,所以foo(NULL);将会调用foo(int),这并不是程序员想要的行为,也违反了代码的直观性。0的歧义在此处造成困扰。

C++11引入了新的关键字来代表空指针常数:nullptr,将空指针和整数0的概念拆开。nullptr的类型为nullptr_t,能隐式转换为任何指针或是成员指针的类型,也能和它们进行相等或不等的比较。而nullptr不能隐式转换为整数,也不能和整数做比较。

为了向下兼容,0仍可代表空指针常数。

1
2
3
4
5
char*pc=nullptr;//OK
int*pi=nullptr;//OK
inti=nullptr;//error

foo(pc);//呼叫foo(char*)
PS:__MINGW32__是MinGW编译器的预定义宏。_MSC_VER是微软C/C++编译器——cl.exe编译代码时预定义的一个宏。_MSC_VER的值表示cl的版本。需要针对cl特定版本编写代码时,也可以使用该宏进行条件编译。

nil和Nil

Objective-C

nil定义为实例对象的空值(anullinstance)

Nil定义为类对象的空值(anullclass)

nil和Nil在objc.h和MacTypes.h文件中均有等价的声明:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
#ifndefNil
#if__has_feature(cxx_nullptr)
#defineNilnullptr
#else
#defineNil__DARWIN_NULL
#endif
#endif
#ifndefnil
#if__has_feature(cxx_nullptr)
#definenilnullptr
#else
#definenil__DARWIN_NULL
#endif
#endif
根据Clang3.7文档对__has_feature的描述:“__has_featureevaluatesto1ifthefeatureisbothsupportedbyClangandstandardizedinthecurrentlanguagestandardor0ifnot”,__has_feature(cxx_nullptr)是用来判断是否支持C++11中的nullptr特性的。在Objective-C中nil和Nil都是__DARWIN_NULL宏定义。按住CMD鼠标点击进入_types.h:

1
2
3
4
5
6
7
8
9
10
11
12
13
#ifdef__cplusplus
#ifdef__GNUG__
#define__DARWIN_NULL__null
#else/*!__GNUG__*/
#ifdef__LP64__
#define__DARWIN_NULL(0L)
#else/*!__LP64__*/
#define__DARWIN_NULL0
#endif/*__LP64__*/
#endif/*__GNUG__*/
#else/*!__cplusplus*/
#define__DARWIN_NULL((void*)0)
#endif/*__cplusplus*/
因为Objective-C不是C++代码,所以倒数第二行#define__DARWIN_NULL((void*)0)此时高亮,意味着最终nil和Nil本质都为((void*)0)

PS:其实如果只看Objective-C中的nil和Nil定义不用这么麻烦的,只需查看Objective-CRuntimeReference中的”Constants->NullValues”即可。

Swift

Swift1.2目前只有nil而没有Nil。为了安全性Swift新增了Optional类型来作为一个容器。好比一个箱子里面可能装有某种类型的对象,也可能是空的(nil)。箱子也可以嵌套,也可以去掉,但这都基于安全的解析、绑定等。Swift的nil和Objective-C中的nil并不一样。在Objective-C中,nil是一个指向不存在对象的指针。在Swift中,nil不是指针——它是一个确定的值,用来表示值缺失。任何类型的可选值都可以被设置为nil,不只是对象(object)类型。

PS:有关Swift中的Optional类型的更多信息可以参考我的另一篇博文:OptionalsandOptionalChaininginSwift

PS:曾几何时,Swift的nil还不是字面量,而是NilType类型的唯一实例。但这一切都是历史了。

NSNull

NSNull在NSNull.h中的定义:

1
2
3
@interfaceNSNull:NSObject<nscopying,nssecurecoding>
+(NSNull*)null;
@end</nscopying,nssecurecoding>
NSNull是个单例,只有一个方法null,也用来表示空值。但它出现在一些nil无法胜任的场景来替代nil来表示空值。比如NSArray和NSDictionary中nil代表数组或字典的末尾(即使nil不出现在末尾,也会将其切断,nil后面的值会丢失),此时只能用NSNull对象来表示空值:

1
2
3
4
NSNull*nullValue=[NSNullnull];
NSArray*arrayWithNull=@[nullValue];
NSLog(@"arrayWithNull:%@",arrayWithNull);
//Output:"arrayWithNull:(<null>)"</null>
虽然NSNull语义上等同于nil,但却并不完全等于nil:

1
2
3
4
5
6
7
8
9
10
11
idaValue=[arrayWithNullobjectAtIndex:0];
if(aValue==nil){
NSLog(@"equalsnil");
}
elseif(aValue==[NSNullnull]){
NSLog(@"equalsNSNullinstance");
if([aValueisEqual:nil]){
NSLog(@"isEqual:nil");
}
}
//Output:"equalsNSNullinstance"
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: