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()
apply,数组
免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
P70系列延期,华为新旗舰将在下月发布
3月20日消息,近期博主@数码闲聊站 透露,原定三月份发布的华为新旗舰P70系列延期发布,预计4月份上市。
而博主@定焦数码 爆料,华为的P70系列在定位上已经超过了Mate60,成为了重要的旗舰系列之一。它肩负着重返影像领域顶尖的使命。那么这次P70会带来哪些令人惊艳的创新呢?
根据目前爆料的消息来看,华为P70系列将推出三个版本,其中P70和P70 Pro采用了三角形的摄像头模组设计,而P70 Art则采用了与上一代P60 Art相似的不规则形状设计。这样的外观是否好看见仁见智,但辨识度绝对拉满。
更新日志
- 雨林唱片《赏》新曲+精选集SACD版[ISO][2.3G]
- 罗大佑与OK男女合唱团.1995-再会吧!素兰【音乐工厂】【WAV+CUE】
- 草蜢.1993-宝贝对不起(国)【宝丽金】【WAV+CUE】
- 杨培安.2009-抒·情(EP)【擎天娱乐】【WAV+CUE】
- 周慧敏《EndlessDream》[WAV+CUE]
- 彭芳《纯色角3》2007[WAV+CUE]
- 江志丰2008-今生为你[豪记][WAV+CUE]
- 罗大佑1994《恋曲2000》音乐工厂[WAV+CUE][1G]
- 群星《一首歌一个故事》赵英俊某些作品重唱企划[FLAC分轨][1G]
- 群星《网易云英文歌曲播放量TOP100》[MP3][1G]
- 方大同.2024-梦想家TheDreamer【赋音乐】【FLAC分轨】
- 李慧珍.2007-爱死了【华谊兄弟】【WAV+CUE】
- 王大文.2019-国际太空站【环球】【FLAC分轨】
- 群星《2022超好听的十倍音质网络歌曲(163)》U盘音乐[WAV分轨][1.1G]
- 童丽《啼笑姻缘》头版限量编号24K金碟[低速原抓WAV+CUE][1.1G]