写出更好的JavaScript程序之undefined篇(中)


Posted in Javascript onNovember 23, 2009

写在前面的依然是消歧义声明:本文中JavaScript是指一般意义上的JavaScript,并不只限定“自称是JavaScript”的运行环境;“全局变量”和“全局对象的属性”是指同样的东西,只是因为要配合上下文才用了不同的说法,正文中我就不再另外解释了;“声明”指通过“var”语句声明变量和/或对函数及其签名的定义;“变量”指通过“var”语句声明过或者在函数体中试图访问的命名参数;“undefined”指名为“undefined”的值(全局或本地变量),而“未定义”指type(...) == “undefined”的概念值;“output”是向我们显示传入参数的函数,其实就是“alert”的同类。

首先介绍的是和void(0)具有异曲同工之妙的一种办法,我们知道在JavaScript中尝试访问任何一个“没有返回值”的函数的执行结果,都会得到“未定义”这个概念值,所以我们可以这样做:
myVar = function(){}();
这个方法的原理是创建一个空函数,并且获取其(根本不存在的)返回值。很明显,我们会得到“未定义”这个概念值,这个原理是和void(0)一样的。
需要特别注意的是,只有当function关键字不在语句之首的时候,才可以使用这种调用方式;如果需要直接调用一个匿名函数,应该这样做:

(function(){
// code here
})();
假如我们不想污染全局作用域,我们就可以用这种方式来创造一个闭包——很多现有的JavaScript库就是这样做的。
这种办法是在不了解void(0)的执行效果的情况下诞生的,既然我们知道void(0),也就不需要这种办法了。

第二种不常见的办法是访问window.undefined,就像下边这样:
output(myVar === window.undefined);
myVar = window.undefined;
这种办法的原理是:
如果JavaScript的运行环境预定义了undefined这一值,window.undefined就可以直接访问到;
如果没有预先定义undefined这一值,window.undefined就会返回“未定义”这一概念值——还是我们想要的东西。
访问形式从“全局变量”变成“属性”以后,脚本引擎就不会认为这是一个意外操作了,因此代码可以正确执行。

然而这种办法还是存在不足:
undefined在所有的JavaScript引擎中都不是保留字,也就是说它是可以被污染的,这会影响我们的程序的运行效果;
直接访问全局对象的属性效率是很差的,应该避免这么做。

因此,接下来就要介绍我所见过的最为有趣的办法——在本地声明undefined变量!
它的做法是这样的:
function myFunc(){
var undefined;
// some code here
output(myVar === undefined);
myVar = undefined;
}
说到这里我要提一下,有的人对“早期的浏览器上没有undefined”这一说法有不同的理解,认为上面这样的做法是行不通的。
而我认为这种观点是不对的,在我的印象中(大约是2003年),IE5虽然没有预定义undefined,但并不影响我们的不指派声明。
不过毕竟是很多年前的事情,记忆模糊,为了避免想当然造成的错误,我特地装了Windows 98来实验一下:
写出更好的JavaScript程序之undefined篇(中)
写出更好的JavaScript程序之undefined篇(中)

上面两幅是用了QQ屏幕截取 ,不知道为什么变成了JPG,下边这幅是用OneNote截取的。

写出更好的JavaScript程序之undefined篇(中)

由此可以得出,为这样不指派声明undefined变量担忧是多余的。
前一篇解释过,在JavaScript中,如果尝试读取一个没有预定义也未声明过的变量,会引发一个“不存在(未定义)”异常,因此在早期的浏览器上如果直接访问undefined的操作可能会失败。
这种(引发异常的)现象可能是为了方便排除故障而有意设计的;而对于声明过的变量,即使是未经赋值,也不会在读取的时候引发异常——经过声明就表示开发人员确定有“这个东西”。

上面这种在函数作用域内声明本地变量undefined的做法,既解决了引用未声明变量会引发异常的问题,又避免了受全局变量污染的影响,还取代了void(0)这种乍看之下意味不明的小把戏,不仅没有造成难以觉察的拼写错误的隐患,还能提高运算效率——真是一举N得的妙招!
所以了解了这么多之后就发现,最后介绍的这种办法实在是居家旅行、杀人灭口的首选
如果有人问:“undefined应该怎么用?”我们现在就可以回答:就这么用!

终于到了结尾,有的人可能忍不住要问了:为什么在这个系列中我的代码示例都用三个等号的“严格相等”来比较?两个等号的普通的“相等”有什么问题吗?
这就是这个系列的下篇要讲解的内容:undefined及其各种等价形态的适用场合。

Javascript 相关文章推荐
JavaScript 模仿vbs中的 DateAdd() 函数的代码
Aug 13 Javascript
指定位置如果有图片显示图片,无图片显示广告的JS
Jun 05 Javascript
js触发asp.net的Button的Onclick事件应用
Feb 02 Javascript
jquery通过visible来判断标签是否显示或隐藏
May 08 Javascript
浅谈Unicode与JavaScript的发展史
Jan 19 Javascript
JavaScript检测浏览器cookie是否已经启动的方法
Feb 27 Javascript
JS+CSS实现大气清新的滑动菜单效果代码
Oct 22 Javascript
Extjs4.0 ComboBox如何实现三级联动
May 11 Javascript
浅谈jQuery效果函数
Sep 16 Javascript
小程序兼容安卓和IOS数据处理问题及坑
Sep 18 Javascript
JavaScript键盘事件常见用法实例分析
Jan 03 Javascript
测量JavaScript函数的性能各种方式对比
Apr 27 Javascript
通过身份证号得到出生日期和性别的js代码
Nov 23 #Javascript
javascript 在网页中的运用(asp.net)
Nov 23 #Javascript
javascript DOM编程实例(智播客学习)
Nov 23 #Javascript
传智播客学习之java 反射
Nov 22 #Javascript
Javascript 刷新全集常用代码
Nov 22 #Javascript
写出更好的JavaScript之undefined篇(上)
Nov 22 #Javascript
COM中获取JavaScript数组大小的代码
Nov 22 #Javascript
You might like
PHP输出缓存ob系列函数详解
2014/03/11 PHP
PHP中的traits简单使用实例
2015/05/13 PHP
php读取csc文件并输出
2015/05/21 PHP
php链表用法实例分析
2015/07/09 PHP
PHPMailer使用QQ邮箱实现邮件发送功能
2017/08/18 PHP
PHP基于DateTime类解决Unix时间戳与日期互转问题【针对1970年前及2038年后时间戳】
2018/06/13 PHP
Ajax一统天下之Dojo整合篇
2007/03/24 Javascript
JavaScript脚本语言在网页中的简单应用
2007/05/13 Javascript
JavaScript 学习小结(适合新手参考)
2009/07/30 Javascript
document.getElementById介绍
2011/09/13 Javascript
7款吸引人眼球的jQuery/CSS3特效实例分享
2013/04/25 Javascript
Firefox中通过JavaScript复制数据到剪贴板(Copy to Clipboard 跨浏览器版)
2013/11/22 Javascript
javascript数组去重方法终极总结
2014/06/05 Javascript
JS点击某个图标或按钮弹出文件选择框的实现代码
2016/09/27 Javascript
详解使用webpack构建多页面应用
2017/12/21 Javascript
微信小程序分享海报生成的实现方法
2018/12/10 Javascript
Vue自定义指令写法与个人理解
2019/02/09 Javascript
vue ssr服务端渲染(小白解惑)
2019/11/10 Javascript
详解Python命令行解析工具Argparse
2016/04/20 Python
Python中的左斜杠、右斜杠(正斜杠和反斜杠)
2016/08/30 Python
python交互式图形编程实例(二)
2017/11/17 Python
Python之web模板应用
2017/12/26 Python
手把手教你python实现SVM算法
2017/12/27 Python
Python数据处理numpy.median的实例讲解
2018/04/02 Python
python自动查询12306余票并发送邮箱提醒脚本
2018/05/21 Python
基于Python实现定时自动给微信好友发送天气预报
2018/10/25 Python
Python 数据库操作 SQLAlchemy的示例代码
2019/02/18 Python
Django使用中间键实现csrf认证详解
2019/07/22 Python
在macOS上搭建python环境的实现方法
2019/08/13 Python
Python基于Dlib的人脸识别系统的实现
2020/02/26 Python
Python3之乱码\xe6\x97\xa0\xe6\xb3\x95处理方式
2020/05/11 Python
Keras搭建自编码器操作
2020/07/03 Python
Python自动巡检H3C交换机实现过程解析
2020/08/14 Python
美国知名的在线旅游服务网站:Priceline
2016/07/23 全球购物
公安交警中队队长个人对照检查材料思想汇报
2014/10/05 职场文书
2015年党员岗位承诺书
2015/04/27 职场文书