详解javascript中原始数据类型Null和Undefined


Posted in Javascript onDecember 17, 2015

当讨论JavaScript中的原始数据类型时,大多数人都知道的基本知识,从String,Number到Boolean。这些原始类型相当简单,行为符合常识。但是,本文将更多聚焦独特的原始数据类型Null和Undefined,是什么让他们如此相似,却又似是而非。

一、理解Null和Undefined
在JavaScript中,null是字面量同时也是语言中的关键字,用来表示无法识别的对象值。换句话说,这用来表示“无值(no value)”。虽然相似,undefined实际上代表了不存在的值(non-existence of a value)。都是完全不可变的,没有属性和方法,也不能给其属性赋值。事实上,试图访问或定义一个属性将会引发一个类型错误(TypeError)。正如他们的名字暗示的那样,他们是完全无效的值。

没有值代表的布尔值是false,这意味着他们在条件上下文中会被被计算为false,如if语句。使用相等操作符(= =)比较这两个值和其他false值,他们并不等于除了自己:

null == 0; // false
undefined == ""; // false
null == false; // false
undefined == false; // false
null == undefined; // true

尽管如此,和其他相似之处,但null和undefined并不是等价的。每个作为其独特的类型的唯一成员,undefined是Undefined类型和null是Null类型。使用全等操作符(===)比较这两个值,这要求类型和值都相等,下面证明这一点:

undefined === null; // false

这是一个重要的区别,服务于不同的目的和理由。区分这两个值,你可以认为undefined代表一个意想不到的没有值而null作为预期没有值的代表。
二、产生Undefined
有许多的方法产生一个undefined值的代码。它通常遇到当试图访问一个不存在的值时。在这种情况下,在JavaScript这种动态的弱类型语言中,只会默认返回一个undefined值,而不是上升为一个错误。
任何声明变量时没有提供一个初始值,都会有一个为undefined的默认值:

var foo; // 默认值为 undefined

当试图访问一个不存在的对象属性或数组项时,返回一个undefined值:

var array = [1, 2, 3];
var foo = array.foo; // foo 属性不存在, 返回 undefined
var item = array[5]; // 数组中没有索引为5的项,返回 undefined

如果省略了函数的返回语句,返回undefined:

var value = (function(){})(); // 返回 undefined

函数调用时未提供的值结果将为undefined参数值:

(function(undefined){
  // 参数是 undefined
})();

void操作符也可以返回一个undefined值。像Underscore的库使用它作为一个防御式的类型检查,因为它是不可变的,可以在任何上下文依赖返回undefined:

function isUndefined(obj){
  return obj === void 0;
}

最后,undefined是一个预定义的全局变量(不像null关键字)初始化为undefined值:

'undefined' in window; // true

ECMAScript 5中,这个变量是只读的,以前并非如此。

三、Null的用例
null的用例是使他与众不同的主要方面,因为不像undefined,null被认为是更有用。这正是为什么typeof操作符作用于null值时返回“object”。最初的理由是,现在仍然是,通常用作一个空引用一个空对象的预期,就像一个占位符。typeof的这种行为已经被确认为一个错误,虽然提出了修正,出于后兼容的目的,这一点已经保持不变。
这就是为什么JavaScript环境从来没有设置一个值为null;它必须以编程方式完成。正如文档MDN所说:
在api中,null是经常检索对象的地方可以预期,但没有相关的对象。
这适用于DOM,它是独立于语言的,不属于ECMAScript规范的范围。因为它是一个外部API,试图获取一个不存在的元素返回一个null值,而不是undefined。
一般来说,如果你需要给一个变量或属性指定一个不变值,将它传递给一个函数,或者从一个函数返回null,null几乎总是最好的选择。简而言之,JavaScript使用undefined并且程序员应该使用null。
null的另一个可行的用例,也被认为是良好的实践是一个显式指定变量为无效(object= null)当一个引用不再是必需的。通过分配null值,有效地清除引用,并假设对象没有引用其他代码,指定垃圾收集,确保回收内存。
四、深入挖掘
使null和undefined像黑洞的不只是他们的行为,而是在他们在JavaScript环境的内部的处理方式。他们似乎通常并不具有同样的关联特征与其他原生或内置对象。
在ES5中,Object.prototype.toString方法,已经成为实际的类型检查标准,这在null和undefined中被证明是一致的:

Object.prototype.toString.call(null); // [object Null]
Object.prototype.toString.call(undefined); // [object Undefined]

然而,Object.prototype.toString方法实际上并不是检索null的内部[[Class]]属性或undefined的公开构造函数。根据文档,以下步骤发生在被调用过程中:

如果值是undefined,返回“[object Undefined]”。

  • 如果这个值为null,则返回“[object Null]”。
  • 让O作为调用ToObject同时传递this值作为参数的结果值。
  • 让class是O的内部属性[[Class]]的值。
  • 返回的结果连接三个字符串“[object ”,class,和“]”的结果的字符串值。

该方法执行一个简单的字符串返回值,如果它检测到null或undefined和其他对象统一的功能。在整个规范中这是很常见的,因为当遇到null和undefined值时大多数方法包含一个简单的捕捉并返回。事实上,没有迹象表明他们包含与任何原生对象相关联的内部属性。就好像他们不是对象。我很想知道如果一个JavaScript的原生环境内部实际存在的显式方案会怎样?也许有人更熟悉一个可以参与的实现。

结论
无论这些原生对象多么不寻常,理解null和undefined之间的差异,和他们在JavaScript的语言基础中截然不同的角色。它可能不能使你的应用程序有所突破,但是一般来说,它仅被证明在开发和调试中长期有益。

以上就是针对javascript中原始数据类型Null和Undefined进行的介绍,希望对大家的学习有所帮助。

Javascript 相关文章推荐
通过身份证号得到出生日期和性别的js代码
Nov 23 Javascript
js模拟C#中List的简单实例
Mar 06 Javascript
对new functionName()定义一个函数的理解
May 22 Javascript
使用javascript实现简单的选项卡切换
Jan 09 Javascript
JavaScript运动框架 解决速度正负取整问题(一)
May 17 Javascript
微信小程序自动客服功能
Nov 02 Javascript
vue自定义全局组件(自定义插件)的用法
Jan 30 Javascript
基于elementUI使用v-model实现经纬度输入的vue组件
May 12 Javascript
node实现简单的增删改查接口实例代码
Aug 22 Javascript
jQuery实现聊天对话框
Feb 08 jQuery
Postman无法正常返回结果问题解决
Aug 28 Javascript
我所理解的JavaScript中的this指向
Sep 04 Javascript
JS实现合并两个数组并去除重复项只留一个的方法
Dec 17 #Javascript
JS数组合并push与concat区别分析
Dec 17 #Javascript
理解Javascript的call、apply
Dec 16 #Javascript
javascript实现网站加入收藏功能
Dec 16 #Javascript
javascript实现无缝上下滚动特效
Dec 16 #Javascript
js实现将选中内容分享到新浪或腾讯微博
Dec 16 #Javascript
JavaScript运行过程中的“预编译阶段”和“执行阶段”
Dec 16 #Javascript
You might like
php利用iframe实现无刷新文件上传功能的代码
2011/09/29 PHP
phpmyadmin打开很慢的解决方法
2014/04/21 PHP
解决cPanel无法安装php5.2.17
2014/06/22 PHP
图片格式的JavaScript和CSS速查手册
2007/08/20 Javascript
一个简单的js鼠标划过切换效果
2010/06/30 Javascript
JQuery each()函数如何优化循环DOM结构的性能
2012/12/10 Javascript
js中parseInt函数浅谈
2013/07/31 Javascript
eclipse导入jquery包后报错的解决方法
2014/02/17 Javascript
JavaScript高级教程5.6之基本包装类型(详细)
2015/11/23 Javascript
jQuery插件扩展实例【添加回调函数】
2016/11/26 Javascript
vue 2.0组件与v-model详解
2017/03/27 Javascript
微信小程序 自动登陆PHP源码实例(源码下载)
2017/05/08 Javascript
原生JavaScript实现精美的淘宝轮播图效果示例【附demo源码下载】
2017/05/27 Javascript
微信小程序中post方法与get方法的封装
2017/09/26 Javascript
详解基于vue-cli3快速发布一个fullpage组件
2019/03/08 Javascript
Layui数据表格判断编辑输入的值,是否为我需要的类型详解
2019/10/26 Javascript
支付宝小程序实现省市区三级联动
2020/06/21 Javascript
解决父组件将子组件作为弹窗调用只执行一次created的问题
2020/07/24 Javascript
[57:18]DOTA2上海特级锦标赛主赛事日 - 1 败者组第一轮#3VP VS VG
2016/03/03 DOTA
python3中bytes和string之间的互相转换
2017/02/09 Python
TensorFlow实现Softmax回归模型
2018/03/09 Python
Python文件如何引入?详解引入Python文件步骤
2018/12/10 Python
python使用pygame模块实现坦克大战游戏
2020/03/25 Python
pyqt弹出新对话框,以及关闭对话框获取数据的实例
2019/06/18 Python
Python OpenCV调用摄像头检测人脸并截图
2020/08/20 Python
对Python 中矩阵或者数组相减的法则详解
2019/08/26 Python
Python代码需要缩进吗
2020/07/01 Python
详解python方法之绑定方法与非绑定方法
2020/08/17 Python
html5使用canvas绘制文字特效
2014/12/15 HTML / CSS
Sephora丝芙兰菲律宾官方网站:购买化妆品和护肤品
2017/04/05 全球购物
英国在线泳装店:Simply Swim
2019/05/05 全球购物
Ruby如何定义一个类
2012/10/08 面试题
管理建议书范文
2014/05/13 职场文书
学校师德承诺书
2014/05/23 职场文书
爱心募捐通知范文
2015/04/27 职场文书
java.util.NoSuchElementException原因及两种解决方法
2022/06/28 Java/Android