[Effective JavaScript 笔记]第54条:将undefined看做“没有值”
undefined值很特殊,每当js无法提供具体的值时,就会产生undefined。
undefined值场景
未赋值的变量的初始值即为undefined。
var x;
x;//undefined
访问对象不存在的属性也会产生undefined。
var obj={};
obj.x;//undefined
一个函数体结尾使用未带参数的return语句,或未使用return语句都会返回值undefined。
function f(){
return;
}
function g(){}
f();//undefined
g();//undefined
未给函数参数提供实参则该参数的值为undefined。
function f(x){
return x;
}
f();//undefined
这几种情况下undefined值表明操作结果并不是一个特定的值。或说叫做“没有值”,但undefined是一个值,很自相矛盾。因为js里的每个操作都要产生点什么,所以就用undefined来填补这些空白了。
undefined公约
将undefined看做缺少某个特定值是js里的公约。将它用于其他的目的,可能会产生严重的问题。例如,一个用户界面元素库可能支持一个highlight方法用于改变一个元素的背景颜色。
element.highlight();//使用默认的颜色
element.highlight('yellow');//使用黄色
如果想提供一种方法来设置一个随机颜色,可能会使用undefined作为特殊的值来实现。
element.highlight(undefined);//使用随机颜色
但上面的代码会产生一个歧义,因为undefined在不传入实参的时候,形参的值也是undefined。所以程序无法在默认颜色,还是随机颜色之间做决定。程序产生了二义性无法正确运行。
再比如,程序可能用一个可选的颜色偏好的配置对象。
var config=JSON.parse(preferences);
//...
element.highlight(config.highlightColor);
如果config.highlightColor并没有设定,那么它的值也会是undefined。这个时候程序员可能希望得到一个默认的颜色。但由于上面的undefined被做为一个特殊的值,这样像上面说的产生了二义性,产生了一个随机的颜色值。更好的方式是对可能会使用一种特殊的颜色名来实现随机颜色。
element.highlight('random');
有时一个API不能够从通常函数可接受的字符串集合中区分出一个特殊的字符串值。在这种情况下,可以使用除undefined以外的其他特殊值,如null或true。这往往会导致可读性下降。
element.highlight(null);
如果阅读代码的人没有记住这个函数的具体使用方法,那么这样的代码很难理解。可能会给人以取消元素高亮的误解。一个更具明确、更具描述性的可选做法是将随机情况表示为一个具有random属性的对象
element.highlight({random:true});
可选参数的实现
undefined有可能出问题的地方是可选参数的实现。理论上arguments对象可检测是否传递了一个参数,但实际上,测试是否为undefined会使API更健壮。例如,一个Web服务器可以接收一个可选的主机名称。
var s1=new Server(80,'example.com');
var s2=new Server(80);//默认为localhost
参数长度
可以通过判断arguments.length来实现Server构造函数。
function Server(port,hostname){
if(arguments.length<2){
hostname='localhost';
}
hostname=''+hostname;
}
上面的这个实现也是会产生问题,当我第二个参数显式传递从其他源请求的一个值。那么可能变成传递的是一个undefined值,但这个时候arguments.length的值是2,代码被破坏。
var s3=new Server(80,config.hostname);
上面代码如果config.hostname没有值,hostname的正常行为应该是设置为'localhost'。上面的代码会产生一个undefined的主机名。
检测参数是否为undefined
最好是测试undefined,不传递参数和传递一个表达式结果是undefined的情况都可以覆盖。
function Server(port,hostname){
if(hostname === undefined){
hostname='localhost';
}
hostname=''+hostname;
//...
}
测试参数是否为真
另一种替代方案是测试hostname是否为真。使用逻辑运算符很好实现
function Server(port,hostname){
hostname=''+(hostname||'localhost');
//...
}
|| 逻辑运算符或。当第一个参数为真值则会返回第一个参数,否则返回第二个参数。所以hostname值为undefined或空字符串,该表达式(hostname||'localhost')的结果是'localhost'。这里把hostname的值测试了很多种情况,包括其他能转化为false的值。这里对于Server函数是可以的。真值测试是实现参数默认值的一种简明的方式。
真值测试
真值测试并不是总是安全的。如果一个函数应该接收空字符串为合法值,真值测试将覆盖空字符串并返回默认值。还有其它的可以转化为false的值,则不应该使用真值测试。例如,一个用于创建用户界面元素的函数可能允许一个元素的宽度或高度为0,但提供默认值却不一样。
var c1=new Element(0,0);//width:0 height:0
var c2=new Element();//width:320 height:240
使用真值测试的实现会出现问题
function Element(w,h){
this.width=w||320;
this.height=h||240;
}
var c1=new Element(0,0);
c1.width;//320
c1.height;//240
这里就不能使用真值而应该使用undefined检测
function Element(w,h){
this.width=w===undefined?320:w;
this.height=h===undefined?240:h;
}
var c1=new Element(0,0);
c1.width;//320
c1.height;//240
var c2=new Element();
c2.width;//320
c2.height;//240
提示
避免使用undefined表示任何非特定值
使用描述性的字符串值或命名布尔属性的对象,而不要使用undefined或null来代表特定应用标志
提供参数默认值应当采用测试undefined的方式,而不是检查arguments.length
在允许0、NaN或空字符串为有效参数的地方,绝不要通过真值测试来实现参数默认值
相关阅读
[Effective JavaScript 笔记]第54条:将undefined看做“没有值”的更多相关文章
- [Effective JavaScript 笔记] 第4条:原始类型优于封闭对象
js有5种原始值类型:布尔值.数字.字符串.null和undefined. 用typeof检测一下: typeof true; //"boolean" typeof 2; //&q ...
- [Effective JavaScript 笔记] 第5条:避免对混合类型使用==运算符
“1.0e0”=={valueOf:function(){return true;}} 是值是多少? 这两个完全不同的值使用==运算符是相等的.为什么呢?请看<[Effective JavaSc ...
- [Effective JavaScript 笔记]第28条:不要信赖函数对象的toString方法
js函数有一个非凡的特性,即将其源代码重现为字符串的能力. (function(x){ return x+1 }).toString();//"function (x){ return x+ ...
- [Effective JavaScript 笔记]第27条:使用闭包而不是字符串来封装代码
函数是一种将代码作为数据结构存储的便利方式,代码之后可以被执行.这使得富有表现力的高阶函数抽象如map和forEach成为可能.它也是js异步I/O方法的核心.与此同时,也可以将代码表示为字符串的形式 ...
- [Effective JavaScript 笔记]第55条:接收关键字参数的选项对象
53节建议保持参数顺序的一致约定对于帮助程序员记住每个参数在函数调用中的意义很重要.参数较少这个主意不错,但如果参数过多后,就出现麻烦了,记忆和理解起来都不太容易. 参数蔓延 如下面这些代码: var ...
- [Effective JavaScript 笔记]第45条:使用hasOwnProperty方法以避免原型污染
之前的43条,44条讨论了属性的枚举,但都没有彻底地解决属性查找中原型污染的问题.看下面关于字典的一些操作 'zhangsan' in dict; dict.zhangsan; dict.zhangs ...
- [Effective JavaScript 笔记]第68条:使用promise模式清洁异步逻辑
构建异步API的一种流行的替代方式是使用promise(有时也被称为deferred或future)模式.已经在本章讨论过的异步API使用回调函数作为参数. downloadAsync('file.t ...
- [Effective JavaScript 笔记]第67条:绝不要同步地调用异步的回调函数
设想有downloadAsync函数的一种变种,它持有一个缓存(实现为一个Dict)来避免多次下载同一个文件.在文件已经被缓存的情况下,立即调用回调函数是最优选择. var cache=new Dic ...
- [Effective JavaScript 笔记] 第6条:了解分号插入的局限
分号可以省略 js可以在语句结束不强制加分号.(建议还是添加,不添加分号往往会出现不易发现的BUG) function Point(x,y){ this.x=x||0; this.y=y||0; } ...
随机推荐
- fis-receiver:一行命令将项目部署到远程服务器
前言 本项目基于FIS2,没了.其实fis项目本身就提供了php版本的范例,这里翻译成node版本. 项目地址:https://github.com/chyingp/fis-receiver 服务端接 ...
- android之外部文件存储和读取
这次借用上次读写内部存储的代码,只是对将更换文件的读写路径即可.这里需要对获取SDcard的读写权限. 一.AndroidManifest.xml 这里增加了对外部存储设备的读写权限 <?xml ...
- javascript继承(三)—继承的实现原理
打算针对js的继承写一系列文章,详细的分析js里继承原理,实现方式,各种继承方式的优缺点,以及最优继承方案,还有多继承的问题等…. 面向对象的编程的核心是封装.继承和多态,js可以看作是一种面向对象的 ...
- android学习——Android Studio下创建menu布局文件
一.问题: android studio项目中没有看到menu文件夹: 在android studio项目中想要添加menu布局文件,一开始我的做法是:直接在res文件夹右键选择xml文件来添加,如下 ...
- 7.Android之评分条RatingBar和拖动条SeekBar学习
评分条RatingBar和拖动条SeekBar很常见,今天来学习下. (1)RatingBar评分条 如图: <RelativeLayout xmlns:android="http:/ ...
- opencv笔记3:trackbar简单使用
time:2015年 10月 03日 星期六 13:54:17 CST # opencv笔记3:trackbar简单使用 当需要测试某变量的一系列取值取值会产生什么结果时,适合用trackbar.看起 ...
- 【SDOI2009】解题汇总
又开了波专题,感觉就和炉石开冒险一样...(说的好像我有金币开冒险似的) /---------------------------------------------/ BZOJ-1226 [SDOI ...
- Docker Architecture、Docker Usage
目录 . 引言 - 为什么要有Docker技术 . Docker简介 . Docker安装.部署.使用 . Docker安全 . Docker底层实现 . Docker网络配置 . Dockerfil ...
- CMD修复
应该命令的路径被修改了. 试下在cmd下打入 path 命令看看.以下是正确的显示. PATH=C:\WINDOWS\system32;C:\WINDOWS;C:\WINDOWS\System32\ ...
- MySql批处理的小窍门:排行榜类数据生成
MySql批处理的小窍门:排行榜类数据生成 最近在做新版本的开发,其中涉及到排行榜的批量预生成,在此分享给大家. 关键点 名次的计算(不考虑用游标) 单榜单查询 对于排行榜这种类型的数据,当只查一个排 ...