当前位置: 首页 > news >正文

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

本文由我们团队的 康祖彬 童鞋撰写,这是他的个人主页: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 *)0C 指针的字面空值
nil(id)0Objective-C 对象的字面空值
Nil(Class)0Objective-C 类的字面空值
NSNull[NSNull null]用来表示空值的 Objective-C 对象

Reference

  • nil / Nil / NULL / NSNull
  • Difference between nil, NiL and, NULL in Objective-C
  • nil/Nil/NULL/NSNull的区别
  • Objective C 中的nil,Nil,NULL和NSNull理解



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

相关文章:

  • Spring 4 官方文档学习(十)数据访问之ORM
  • sofa-pbrpc高级用法
  • combination-sum-ii(熟悉下Java排序)
  • hping网络安全工具的安装及使用
  • Android开发学习——应用安装过程
  • class path resource [applicationContext.xml] cannot be opened because it does not exis
  • Arrar.prototype.map()的用法
  • [ERROR] Plugin 'InnoDB' init function returned error
  • oracle授权动态视图权限给用户
  • RAC 11.2的新特性
  • ABP文档 - Javascript Api
  • node学习第三天(2)
  • Ansible之玩转常见运维场景(个人总结)
  • 照虎画虎——简单WeUI模板UX设计学习
  • There is no Action mapped for namespace [/] and action name [TestAction] ass
  • 《微软的软件测试之道》成书始末、出版宣告、补充致谢名单及相关信息
  • 【划重点】MySQL技术内幕:InnoDB存储引擎
  • 【跃迁之路】【477天】刻意练习系列236(2018.05.28)
  • 2017年终总结、随想
  • 4月23日世界读书日 网络营销论坛推荐《正在爆发的营销革命》
  • CentOS学习笔记 - 12. Nginx搭建Centos7.5远程repo
  • Cookie 在前端中的实践
  • CSS进阶篇--用CSS开启硬件加速来提高网站性能
  • gops —— Go 程序诊断分析工具
  • Javascript基础之Array数组API
  • Java多态
  • Java-详解HashMap
  • js对象的深浅拷贝
  • Python - 闭包Closure
  • Python 使用 Tornado 框架实现 WebHook 自动部署 Git 项目
  • Vue ES6 Jade Scss Webpack Gulp
  • 得到一个数组中任意X个元素的所有组合 即C(n,m)
  • 读懂package.json -- 依赖管理
  • 观察者模式实现非直接耦合
  • 开年巨制!千人千面回放技术让你“看到”Flutter用户侧问题
  • PostgreSQL之连接数修改
  • 资深实践篇 | 基于Kubernetes 1.61的Kubernetes Scheduler 调度详解 ...
  • ​LeetCode解法汇总307. 区域和检索 - 数组可修改
  • #NOIP 2014# day.1 生活大爆炸版 石头剪刀布
  • #微信小程序(布局、渲染层基础知识)
  • (2)nginx 安装、启停
  • (C语言)fread与fwrite详解
  • (LeetCode C++)盛最多水的容器
  • (附源码)ssm户外用品商城 毕业设计 112346
  • (亲测)设​置​m​y​e​c​l​i​p​s​e​打​开​默​认​工​作​空​间...
  • (十五)Flask覆写wsgi_app函数实现自定义中间件
  • (算法)前K大的和
  • (转)Android学习笔记 --- android任务栈和启动模式
  • .htaccess配置重写url引擎
  • .NET delegate 委托 、 Event 事件,接口回调
  • .Net Framework 4.x 程序到底运行在哪个 CLR 版本之上
  • .NET Standard / dotnet-core / net472 —— .NET 究竟应该如何大小写?
  • .NET/C# 使用 ConditionalWeakTable 附加字段(CLR 版本的附加属性,也可用用来当作弱引用字典 WeakDictionary)
  • @Transactional 详解
  • [AIGC] 如何建立和优化你的工作流?