前言
记得曾经有不少朋友问过笔者,在Objective-C中nil和Nil以及NULL的区别。最重要的是,在面试中还有不少朋友常会被问到。记得当年刚找工作的时候,笔者就被面试官问到过,现在笔者在这里统一详细说明。
NULL
对于学习过C/C++语言的朋友,对NULL一定很熟悉吧?这就是在C/C++中的空指针。
在C语言中,NULL是无类型的,只是一个宏,它代表空。我们不研究C++中的NULL,因为在C++11以后又有了新的定义,我们不深究。
这就是C语言中所谓的NULL(C++的定义比较复杂,这里不说了):
1
2
3
4
5
6
7
8
9
10
11
12
13
#if defined(__need_NULL)
#undef NULL
#ifdef __cplusplus
# if !defined(__MINGW32__) && !defined(_MSC_VER)
# define NULL __null
# else
# define NULL 0
# endif
#else
# define NULL ((void*)0)
#endif
这是在stddef.h头文件中声明的。这是使用了条件编译的,__cplusplus这个宏表示C++,对于我们Objective-C开发来说,NULL就表示((void*)0)
像C语言中,我们定义了一个指针,当我们使用完以后,通常会设置指向NULL。如果没有设置,这个指针就成了所谓的野指针,然后其它地方不小心访问了这个指针是很容易造成非法访问的,常见的表现就是崩溃了。
id="iframe2456100_0" src="http://pos.baidu.com/zcom?sz=760x90&rtbid=2385975&rdid=10090296&dc=2&di=2456100&dri=0&dis=0&dai=2&ps=1126x358&dcb=BAIDU_SSP_define&dtm=BAIDU_DUP_SETJSONADSLOT&dvi=0.0&dci=-1&dpt=none&tsr=0&tpr=1459786513781&ti=nil%E3%80%81Nil%E3%80%81NULL%E5%92%8CNSNull%E5%8C%BA%E5%88%AB%20%7C%20%E6%A0%87%E5%93%A5%E7%9A%84%E6%8A%80%E6%9C%AF%E5%8D%9A%E5%AE%A2&ari=1&dbv=2&drs=1&pcs=1271x701&pss=1271x1136&cfv=18&cpl=5&chi=1&cce=true&cec=UTF-8&tlm=1459786513<u=http%3A%2F%2Fwww.henishuo.com%2Fnil-nil-null-nsnull-difference%2F<r=https%3A%2F%2Fwww.baidu.com%2Flink%3Furl%3DB_AIGnw-ddqVEVYc237zzo-85qN7oZvIQMSuopJWPfXEKay-W7jpIzR_MMxWdkojA48GBJRWKbM-1-ikEwyxFYvCZchHIJB6zfUumrWepGe%26wd%3D%26eqid%3Dc4a10d6d0000649d0000000357029265&lcr=https%3A%2F%2Fwww.baidu.com%2Flink%3Furl%3DB_AIGnw-ddqVEVYc237zzo-85qN7oZvIQMSuopJWPfXEKay-W7jpIzR_MMxWdkojA48GBJRWKbM-1-ikEwyxFYvCZchHIJB6zfUumrWepGe%26wd%3D%26eqid%3Dc4a10d6d0000649d0000000357029265&ecd=1&psr=1280x800&par=1280x773&pis=-1x-1&ccd=24&cja=true&cmi=7&col=zh-CN&cdo=-1&tcn=1459786514&qn=9e639be37ab13500&dpv=ab9aee502a70f0f8&tt=1459786513697.238.260.268" width="760" height="90" align="center,center" vspace="0" hspace="0" marginwidth="0" marginheight="0" scrolling="no" frameborder="0" allowtransparency="true" style="border-width: 0px; border-style: initial; font-family: inherit; font-style: inherit; font-weight: inherit; margin: 0px; outline: 0px; padding: 0px; vertical-align: bottom; max-width: 100%;">
既然Objective-C是基于C语言的面向对象语言,那么也会使用到C语言类型的指针,比如使用const char *类型,判断是否为空时,是使用p != NULL来判断的。
nil
对于我们学习Objective-C的人来说,这个是非常熟悉的。如下为官方定义:
1
2
3
4
5
6
7
8
9
#ifndef nil
# if __has_feature(cxx_nullptr)
# define nil nullptr
# else
# define nil __DARWIN_NULL
# endif
#endif
对于我们Objective-C开发来说,nil就是__DARWIN_NULL。看下官方定义:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#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 */
这个也是条件编译的,那么对于我们Objective-C开发来说,nil就代表((void *)0)。
我们使用nil表示Objective-C对象为空,如NSString *str = nil。
Nil
先看看官方是如何声明的:
1
2
3
4
5
6
7
8
9
#ifndef Nil
# if __has_feature(cxx_nullptr)
# define Nil nullptr
# else
# define Nil __DARWIN_NULL
# endif
#endif
根据条件,我们做Objective-C开发的,那么Nil也就是代表__DARWIN_NULL,而对于__DARWIN_NULL的声明如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#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 */
这个也是条件编译的,那么对于我们Objective-C开发来说,Nil也就代表((void *)0)。
但是它是用于代表空类的。比如:
1
2
3
Class
myClass
=
Nil
;
NSNull
先看看官方的声明:
id="iframe2457199_0" src="http://pos.baidu.com/zcom?sz=760x90&rtbid=2386521&rdid=10091131&dc=2&di=2457199&dri=0&dis=0&dai=3&ps=3167x358&dcb=BAIDU_SSP_define&dtm=BAIDU_DUP_SETJSONADSLOT&dvi=0.0&dci=-1&dpt=none&tsr=0&tpr=1459786513781&ti=nil%E3%80%81Nil%E3%80%81NULL%E5%92%8CNSNull%E5%8C%BA%E5%88%AB%20%7C%20%E6%A0%87%E5%93%A5%E7%9A%84%E6%8A%80%E6%9C%AF%E5%8D%9A%E5%AE%A2&ari=1&dbv=2&drs=1&pcs=1271x701&pss=1271x3177&cfv=18&cpl=5&chi=1&cce=true&cec=UTF-8&tlm=1459786513<u=http%3A%2F%2Fwww.henishuo.com%2Fnil-nil-null-nsnull-difference%2F<r=https%3A%2F%2Fwww.baidu.com%2Flink%3Furl%3DB_AIGnw-ddqVEVYc237zzo-85qN7oZvIQMSuopJWPfXEKay-W7jpIzR_MMxWdkojA48GBJRWKbM-1-ikEwyxFYvCZchHIJB6zfUumrWepGe%26wd%3D%26eqid%3Dc4a10d6d0000649d0000000357029265&lcr=https%3A%2F%2Fwww.baidu.com%2Flink%3Furl%3DB_AIGnw-ddqVEVYc237zzo-85qN7oZvIQMSuopJWPfXEKay-W7jpIzR_MMxWdkojA48GBJRWKbM-1-ikEwyxFYvCZchHIJB6zfUumrWepGe%26wd%3D%26eqid%3Dc4a10d6d0000649d0000000357029265&ecd=1&psr=1280x800&par=1280x773&pis=-1x-1&ccd=24&cja=true&cmi=7&col=zh-CN&cdo=-1&tcn=1459786514&qn=ca62984a30ea456b&dpv=9dd25309a8715147&tt=1459786513697.287.311.312" width="760" height="90" align="center,center" vspace="0" hspace="0" marginwidth="0" marginheight="0" scrolling="no" frameborder="0" allowtransparency="true" style="border-width: 0px; border-style: initial; font-family: inherit; font-style: inherit; font-weight: inherit; margin: 0px; outline: 0px; padding: 0px; vertical-align: bottom; max-width: 100%;">
1
2
3
4
5
6
7
8
9
10
11
NS_ASSUME_NONNULL_BEGIN
@interface
NSNull
: NSObject
<
NSCopying
,
NSSecureCoding
>
+
(
NSNull
*
)
null
;
@end
NS_ASSUME_NONNULL_END
由此我们可知,NSNull是继承于NSObject的类型。它是很特殊的类,它表示是空,什么也不存储,但是它却是对象,只是一个占位对象。
使用场景就不一样了,比如说服务端接口中让我们在值为空时,传空。
1
2
3
4
NSDictionry
*parameters
=
@
{
@"arg1"
:
@"value1"
,
@"arg2"
: arg2
.
isEmpty
?
[
NSNull
null
]
: arg2
}
;
这只是随手举的例子,当然我们也可以不传这人参数。如果我们要统一,比如通过runtime来动态将对象转成我们的参数时,那么可以统一将值为nil的都设置为[NSNull null]
区别
NULL、nil、Nil这三者对于Objective-C中值是一样的,都是(void *)0,那么为什么要区分呢?又与NSNull之间有什么区别:
NULL是宏,是对于C语言指针而使用的,表示空指针 nil是宏,是对于Objective-C中的对象而使用的,表示对象为空 Nil是宏,是对于Objective-C中的类而使用的,表示类指向空 NSNull是类类型,是用于表示空的占位对象,与JS或者服务端的null类似的含意