使用apply方法处理数组的三个技巧[译]


Posted in Javascript onSeptember 20, 2012

apply方法

apply是所有函数都有的方法.它的签名如下:
func.apply(thisValue, [arg1, arg2, ...])
如果不考虑thisValue的影响,上面的调用等同于:
func(arg1, arg2, ...)
也就是说,apply允许我们将一个数组"解开"成为一个个的参数再传递给调用函数.让我们分别看看apply使用中的三个技巧.

技巧1: 将一个数组传递给一个不接受数组作为参数的函数

JavaScript中没有返回一个数组中最大值的函数.但是,有一个函数Math.max可以返回任意多个数值类型的参数中的最大值.再配合apply,我们可以实现我们的目的:

> Math.max.apply(null, [10, -1, 5]) 
10

译者注:注意Math.max方法的参数中只要有一个值被转为NaN,则该方法直接返回NaN
>Math.max(1,null) //相当于Math.max(1,0) 
1 
>Math.max(1,undefinded) //相当于Math.max(1,NaN) 
NaN >Math.max(0,-0) //正零比负零大,和==不同 
0 
>Math.max(-0,-1) //负零比-1大 
-0

技巧2: 填补稀疏数组

数组中的缝隙
这里提醒一下读者:在JavaScript中,一个数组就是一个数字到值的映射.所以如果某个索引处缺失了一个元素(一条缝隙)和某个元素的值为undefined,是两种不同的情况.前者在被Array.prototype中的相关方法(forEach, map, 等.)遍历时,会跳过那些缺失的元素,而后者不会:

> ["a",,"b"].forEach(function (x) { console.log(x) }) 
a 
> ["a",undefined,"b"].forEach(function (x) { console.log(x) }) 
a 
undefined

译者注:这里作者说"数组就是一个数字到值的映射",严格意义上是不对的,正确的说法是"数组就是一个字符串到值的映射".下面是证据:
>for (i in ["a", "b"]) { 
console.log(typeof i) //数组的索引实际上是个字符串 
} 
"string" 
"string" >["a", "b"].forEach(function (x, i) { 
console.log(typeof i) //这里的i实际上不是索引,只是个数字类型的累加器 
}) 
"number" 
"number"

你可以使用in运算符来检测数组中是否有缝隙.
> 1 in ["a",,"b"] 
false 
> 1 in ["a", undefined, "b"] 
true

译者注:这里之所以用1可以,是因为in运算符会把1转换成"1".

你过你尝试读取这个缝隙的值,会返回undefined,和实际的undefined元素是一样.

> ["a",,"b"][1] 
undefined 
> ["a", undefined, "b"][1] 
undefined

译者注:[1]也会被转换成["1"]

填补缝隙

apply配合Array(这里不需要加new)使用,可以将数组中的缝隙填补为undefined元素:

> Array.apply(null, ["a",,"b"]) 
[ 'a', undefined, 'b' ]

这都是因为apply不会忽略数组中的缝隙,会把缝隙作为undefined参数传递给函数:

> function returnArgs() { return [].slice.call(arguments) } 
> returnArgs.apply(null, ["a",,"b"]) 
[ 'a', undefined, 'b' ]

但需要注意的是,如果Array方法接收到的参数是一个单独的数字,则会把这个参数当成数组长度,返回一个新数组:
> Array.apply(null, [ 3 ]) 
[ , , ]

因此,最靠谱的方法是写一个这样的函数来做这种工作:
function fillHoles(arr) { 
var result = []; 
for(var i=0; i < arr.length; i++) { 
result[i] = arr[i]; 
} 
return result; 
}

执行:
> fillHoles(["a",,"b"]) 
[ 'a', undefined, 'b' ]

Underscore中的_.compact函数会移除数组中的所有假值,包括缝隙:

> _.compact(["a",,"b"]) 
[ 'a', 'b' ] 
> _.compact(["a", undefined, "b"]) 
[ 'a', 'b' ] 
> _.compact(["a", false, "b"]) 
[ 'a', 'b' ]

技巧3: 扁平化数组

任务:将一个包含多个数组元素的数组转换为一个一阶数组.我们利用apply解包数组的能力配合concat来做这件事:

> Array.prototype.concat.apply([], [["a"], ["b"]]) 
[ 'a', 'b' ]

混合非数组类型的元素也可以:
> Array.prototype.concat.apply([], [["a"], "b"]) 
[ 'a', 'b' ]

apply方法的thisValue必须指定为[],因为concat是一个数组的方法,不是一个独立的函数.这种写法的限制是最多只能扁平化二阶数组:
> Array.prototype.concat.apply([], [[["a"]], ["b"]]) 
[ [ 'a' ], 'b' ]

所以你应该考虑一个替代方案.比如Underscore中的_.flatten函数就可以处理任意层数的嵌套数组

> _.flatten([[["a"]], ["b"]]) 
[ 'a', 'b' ]

参考
JavaScript: sparse arrays vs. dense arrays

ECMAScript.next: Array.from() and Array.of()

Javascript 相关文章推荐
Javascript attachEvent传递参数的办法
Dec 14 Javascript
JSON 和 JavaScript eval使用说明
Jun 13 Javascript
Whatever:hover 无需javascript让IE支持丰富伪类
Jun 29 Javascript
提升你网站水平的jQuery插件集合推荐
Apr 19 Javascript
javascript自适应宽度的瀑布流实现思路
Feb 20 Javascript
js判断浏览器版本以及浏览器内核的方法
Jan 20 Javascript
基于JQuery的$.ajax方法进行异步请求导致页面闪烁的解决办法
May 10 Javascript
node实现简单的反向代理服务器
Jul 26 Javascript
vue中post请求以a=a&amp;b=b 的格式写遇到的问题
Apr 27 Javascript
vue watch深度监听对象实现数据联动效果
Aug 16 Javascript
javascript实现一款好看的秒表计时器
Sep 05 Javascript
JavaScript实现一键复制内容剪贴板
Jul 23 Javascript
js DOM 元素ID就是全局变量
Sep 20 #Javascript
JavaScript NaN和Infinity特殊值 [译]
Sep 20 #Javascript
JavaScript 更严格的相等 [译]
Sep 20 #Javascript
JavaScript 反科里化 this [译]
Sep 20 #Javascript
Array.prototype.concat不是通用方法反驳[译]
Sep 20 #Javascript
JavaScript 用Node.js写Shell脚本[译]
Sep 20 #Javascript
一个简单的网站访问JS计数器 刷新1次加1次访问
Sep 20 #Javascript
You might like
无法在发生错误时创建会话,请检查 PHP 或网站服务器日志,并正确配置 PHP 安装最快的解决办法
2010/08/01 PHP
ThinkPHP中的常用查询语言汇总
2014/08/22 PHP
php array_merge函数使用需要注意的一个问题
2015/03/30 PHP
学习php设计模式 php实现原型模式(prototype)
2015/12/07 PHP
Laravel 5使用Laravel Excel实现Excel/CSV文件导入导出的功能详解
2017/10/11 PHP
JS获得浏览器版本和操作系统版本的例子
2014/05/13 Javascript
Node.js中使用mongoskin操作mongoDB实例
2014/09/28 Javascript
js实现网站最上边可关闭的浮动广告条代码
2015/09/04 Javascript
深入理解JavaScript定时机制
2016/10/27 Javascript
基于jQuery和Bootstrap框架实现仿知乎前端动态列表效果
2016/11/09 Javascript
详解vue.js 开发环境搭建最简单攻略
2017/06/12 Javascript
ES6中的rest参数与扩展运算符详解
2017/07/18 Javascript
前端主流框架vue学习笔记第二篇
2017/07/26 Javascript
利用SpringMVC过滤器解决vue跨域请求的问题
2018/02/10 Javascript
微信小程序上传图片功能(附后端代码)
2020/06/19 Javascript
bootstrap下拉框动态赋值方法
2018/08/10 Javascript
NodeJS使用Range请求实现下载功能的方法示例
2018/10/12 NodeJs
VUE解决微信签名及SPA微信invalid signature问题(完美处理)
2019/03/29 Javascript
学习LayUI时自研的表单参数校验框架案例分析
2019/07/29 Javascript
在vue-cli创建的项目中使用sass操作
2020/08/10 Javascript
原生JavaScript实现弹幕组件的示例代码
2020/10/12 Javascript
vue图片裁剪插件vue-cropper使用方法详解
2020/12/16 Vue.js
布同自制Python函数帮助查询小工具
2011/03/13 Python
详解Open Folder as PyCharm Project怎么添加的方法
2020/12/29 Python
使用css3做0.5px的细线的示例代码
2018/01/18 HTML / CSS
html5唤起app的方法
2017/11/30 HTML / CSS
台湾团购、宅配和优惠券:17Life
2017/08/14 全球购物
英文版网络工程师求职信
2013/10/28 职场文书
园林专业毕业生自荐信
2014/07/04 职场文书
普通党员对照检查材料
2014/09/24 职场文书
机票销售员态度不好检讨书
2014/09/27 职场文书
班子成员四风问题自我剖析材料
2014/09/29 职场文书
2014年就业工作总结
2014/11/26 职场文书
三好学生事迹材料
2014/12/24 职场文书
唐山大地震的观后感
2015/06/05 职场文书
家庭教育培训学习心得体会
2016/01/14 职场文书