本文由我们团队的 康祖彬 童鞋撰写,这是他的个人主页:https://kangzubin.cn


理解”不存在“的概念不仅仅是一个哲学的问题,也是一个实际的问题。我们是有形宇宙的居民,而原因在于逻辑宇宙的存在不确定性。作为一个逻辑系统的物理体现,计算机面临一个棘手的问题,就是如何用”存在“表达”不存在“。--摘自 NSHipster

这段话读起来怪怪的,毕竟是翻译过来的,大概意思是说在计算机中如何描述”不存在“这个概念很重要。

在 C 语言中用 0 来作为“不存在”的原始值,而用 NULL 作为指针空值。在 Objective-C 中,则有几种不同的方式来表示“不存在”,分别有:NULLnilNilNSNull。下面我们来看看这几种空值的定义以及使用上的不同。

注:以下各种空值定义的源码摘自 iOS 10.0 SDK 中的相关头文件。

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++ 环境下不同编译器的 __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 <Foundation/NSObject.h>

NS_ASSUME_NONNULL_BEGIN

@interface NSNull : NSObject <NSCopying, NSSecureCoding>

+ (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 中不存在这两个符号!!!

总结

从上述分析我们可知,不管是 NULLnil 还是 Nil,它们本质上是一样的,都是 (void *)0,只是写法不同。这样做的意义是为了区分不同的数据类型,虽然它们值相同,但我们需要理解它们之间的字面意义并用于不同场景,让代码更加明确,增加可读性。

标志 含义
NULL (void *)0 C 指针的字面空值
nil (id)0 Objective-C 对象的字面空值
Nil (Class)0 Objective-C 类的字面空值
NSNull [NSNull null] 用来表示空值的 Objective-C 对象

Reference

文/XcodeMen(简书作者)
原文链接:http://www.jianshu.com/p/5d7033b15052
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

Objective-C 中 NULL、nil、Nil、NSNull 的定义及不同的更多相关文章

  1. Objective C中nil/Nil/NULL的区别

    nil:指向oc中对象的空指针 Nil:指向oc中类的空指针 NULL:指向其他类型的空指针,如一个c类型的内存指针 NSNull:在集合对象中,表示空值的对象 若obj为nil:[obj messa ...

  2. Objective C 中的nil,Nil,NULL和NSNull理解

    kenyo网友的原创说法是:做IOS开发的估计都对Objective-C的内存管理机制很头疼,一不小心程序就会出内存泄露,我也不例外,前几天被指针的置nil与release给搞惨了,今和大家详细解说一 ...

  3. Objective-C 中nil/Nil/NULL/NSNull

    转自:http://nshipster.cn/nil/ 理解"不存在"的概念不仅仅是一个哲学的问题,也是一个实际的问题.我们是有形宇宙的居民,而原因在于逻辑宇宙的存在不确定性.作为 ...

  4. iOS中nil 、NULL、 Nil 、NSNull

    nil,定义一个空的实例,指向OC中对象的空指针. 示例代码: NSString *someString = nil; NSURL *someURL = nil; id someObject = ni ...

  5. OC中Nil nil NULL 和 [NSNULL null]的区别

    关于这个问题看过两三次了,但是每次过两个月脑袋里又会不清晰,索性记录一下加深一下印象. 一.nil 当一个对象置为nil时,这个对象的内存地址就会被系统收回.置空之后是不能进行retain,copy等 ...

  6. nil / Nil / NULL / NSNull VS objc_msgSend

    [NSNull null]是一个对象,其类为NSNULL(isa):里面没有任何变量.函数.和实现. nil的处理展示出消息机制的优越性,相对于函数调用的空指针处理. ENTRY objc_msgSe ...

  7. IOS中nil/Nil/NULL的区别

    类与对象的概念 类是对同一类事物高度的抽象,类中定义了这一类对象所应具有的静态属性(属性)和动态属性(方法). 对象是类的一个实例,是一个具体的事物. 类与对象是抽象与具体的关系. 类其实就是一种数据 ...

  8. IOS 学习笔记 2015-03-20 O之 nil,Nil,NULL,NSNull

    1.oc最好 用nil   [ nil  任意方法],不会崩溃 nil 是一个对象值.NULL是一个通用指针(泛型指针). 2. NSNULL,NULL和nil在本质上应该是一样的,NULL和nil其 ...

  9. objective-C nil,Nil,NULL 和NSNull的小结

    nil用来给对象赋值(Object-C的任何对象都属于id类型),NULL则给任何指针赋值,NULL和nil不能互换,nil用于类指针赋值(在Object-C中类是一个对象,是类的meta-class ...

随机推荐

  1. Checking For User Permissions Before Updating or Inserting The Records in Oracle Forms

    Suppose you want to check the user permissions on inserting or updating the records in Oracle Forms, ...

  2. How To Tune or Test PLSQL Code Performance in Oracle D2k Forms

    You can test or tune your program unit performance in Oracle forms with Ora_Prof package.Suppose you ...

  3. QT 加载c语言编译的动态库

    QLibrary lib("./libprint.so");//库的路径if(lib.load()){    typedef void(*AddFunction)(char *st ...

  4. 屏幕 Dynpro

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  5. \n ^ \t的使用

    \n是换行符: 使用时需要放在字符串里面,成为字符串的一部分(不要嫌乱,这是我之前犯的错误). 示例: >>>print("I am Guido \n what's yon ...

  6. 异步上传图片,光用jquery不行,得用jquery.form.js插件

    异步上传图片,光用jquery不行,得用jquery.form.js插件,百度一下下载这个插件,加jquery,引入就可以了 <form id="postbackground" ...

  7. struts2 传递数组、List、Map

    struts2 传递数组.List.Map jsp文件 数组:     <s:textfield name="ages" value="a1">&l ...

  8. 安装64位版Oracle11gR2后无法启动SQLDeveloper的解决方案(原创) (2016-10-29 下午01:56)

    安装64位版Oracle11gR2后发现启动SQL Developer时弹出配置java.exe的路径,找到Oracle自带java.exe后产生的路径"C:\app\用户名\product ...

  9. 使用xml方式定义补间动画

    在res下创建一个目录 anim目录 public class MainActivity extends Activity { private ImageView iv; @Override prot ...

  10. C 语言预处理

    在嵌入式系统编程中,不管是内核的驱动程序还是应用程序的编写,都涉及到大量的预处理与条件编译,这样做的好处主要体现在代码的移植性强以及代码的修改方便等特性,因此引入了预处理与条件编译的概念.在C语言的程 ...