javascript 遍历
数组的遍历你都会用了,那Promise版本的呢
这里指的遍历方法包括:
map
、reduce
、reduceRight
、forEach
、filter
、some
、every
因为最近要进行了一些数据汇总,node
版本已经是8.11.1了,所以直接写了个async/await
的脚本。
但是在对数组进行一些遍历操作时,发现有些遍历方法对Promise
的反馈并不是我们想要的结果。
当然,有些严格来讲并不能算是遍历,比如说some
,every
这些的。
但确实,这些都会根据我们数组的元素来进行多次的调用传入的回调。
这些方法都是比较常见的,但是当你的回调函数是一个Promise
时,一切都变了。
前言
async/await
为Promise
的语法糖
文中会直接使用async/await
替换Promise
let result = await func()
// => 等价于
func().then(result => {
// code here
})
// ======
async function func () {
return 1
}
// => 等价与
function func () {
return new Promise(resolve => resolve(1))
}
map
map
可以说是对Promise
最友好的一个函数了。
我们都知道,map
接收两个参数:
- 对每项元素执行的回调,回调结果的返回值将作为该数组中相应下标的元素
- 一个可选的回调函数
this
指向的参数
[1, 2, 3].map(item => item ** 2) // 对数组元素进行求平方
// > [1, 4, 9]
上边是一个普通的map
执行,但是当我们的一些计算操作变为异步的:
[1, 2, 3].map(async item => item ** 2) // 对数组元素进行求平方
// > [Promise, Promise, Promise]
这时候,我们获取到的返回值其实就是一个由Promise
函数组成的数组了。
所以为什么上边说map
函数为最友好的,因为我们知道,Promise
有一个函数为Promise.all
会将一个由Promise
组成的数组依次执行,并返回一个Promise
对象,该对象的结果为数组产生的结果集。
await Promise.all([1, 2, 3].map(async item => item ** 2))
// > [1, 4, 9]
首先使用Promise.all
对数组进行包装,然后用await
获取结果。
reduce/reduceRight
reduce
的函数签名想必大家也很熟悉了,接收两个参数:
对每一项元素执行的回调函数,返回值将被累加到下次函数调用中,回调函数的签名:
accumulator
累加的值currentValue
当前正在处理的元素currentIndex
当前正在处理的元素下标array
调用reduce
的数组
- 可选的初始化的值,将作为
accumulator
的初始值
[1, 2, 3].reduce((accumulator, item) => accumulator + item, 0) // 进行加和
// > 6
这个代码也是没毛病的,同样如果我们加和的操作也是个异步的:
[1, 2, 3].reduce(async (accumulator, item) => accumulator + item, 0) // 进行加和
// > Promise {<resolved>: "[object Promise]3"}
这个结果返回的就会很诡异了,我们在回看上边的reduce
的函数签名
对每一项元素执行的回调函数,返回值将被累加到下次函数调用中
然后我们再来看代码,async (accumulator, item) => accumulator += item
这个在最开始也提到了,是Pormise
的语法糖,为了看得更清晰,我们可以这样写:
(accumulator, item) => new Promise(resolve =>
resolve(accumulator += item)
)
也就是说,我们reduce
的回调函数返回值其实就是一个Promise
对象
然后我们对Promise
对象进行+=
操作,得到那样怪异的返回值也就很合情合理了。
当然,reduce
的调整也是很轻松的:
await [1, 2, 3].reduce(async (accumulator, item) => await accumulator + item, 0)
// > 6
我们对accumulator
调用await
,然后再与当前item
进行加和,在最后我们的reduce
返回值也一定是一个Promise
,所以我们在最外边也添加await
的字样
也就是说我们每次reduce
都会返回一个新的Promise
对象,在对象内部都会获取上次Promise
的结果。
我们调用reduce
实际上得到的是类似这样的一个Promise
对象:
new Promise(resolve => {
let item = 3
new Promise(resolve => {
let item = 2
new Promise(resolve => {
let item = 1
Promise.resolve(0).then(result => resolve(item + result))
}).then(result => resolve(item + result))
}).then(result => resolve(item + result))
})
reduceRight
这个就没什么好说的了。。跟reduce
只是执行顺序相反而已
forEach
forEach
,这个应该是用得最多的遍历方法了,对应的函数签名:
callback
,对每一个元素进行调用的函数currentValue
,当前元素index
,当前元素下标array
,调用forEach
的数组引用
thisArg
,一个可选的回调函数this
指向
我们有如下的操作:
// 获取数组元素求平方后的值
[1, 2, 3].forEach(item => {
console.log(item ** 2)
})
// > 1
// > 4
// > 9
普通版本我们是可以直接这么输出的,但是如果遇到了Promise
// 获取数组元素求平方后的值
[1, 2, 3].forEach(async item => {
console.log(item ** 2)
})
// > nothing
forEach
并不关心回调函数的返回值,所以forEach
只是执行了三个会返回Promise
的函数
所以如果我们想要得到想要的效果,只能够自己进行增强对象属性了:
Array.prototype.forEachSync = async function (callback, thisArg) {
for (let [index, item] of Object.entries(this)) {
await callback(item, index, this)
}
}
await [1, 2, 3].forEachSync(async item => {
console.log(item ** 2)
})
// > 1
// > 4
// > 9
await
会忽略非Promise
值,await 0
、await undefined
与普通代码无异
filter
filter
作为一个筛选数组用的函数,同样具有遍历的功能:
函数签名同forEach
,但是callback
返回值为true
的元素将被放到filter
函数返回值中去。
我们要进行一个奇数的筛选,所以我们这么写:
[1, 2, 3].filter(item => item % 2 !== 0)
// > [1, 3]
然后我们改为Promise
版本:
[1, 2, 3].filter(async item => item % 2 !== 0)
// > [1, 2, 3]
这会导致我们的筛选功能失效,因为filter
的返回值匹配不是完全相等的匹配,只要是返回值能转换为true
,就会被认定为通过筛选。Promise
对象必然是true
的,所以筛选失效。
所以我们的处理方式与上边的forEach
类似,同样需要自己进行对象增强
但我们这里直接选择一个取巧的方式:
Array.prototype.filterSync = async function (callback, thisArg) {
let filterResult = await Promise.all(this.map(callback))
// > [true, false, true]
return this.filter((_, index) => filterResult[index])
}
await [1, 2, 3].filterSync(item => item % 2 !== 0)
我们可以直接在内部调用map
方法,因为我们知道map
会将所有的返回值返回为一个新的数组。
这也就意味着,我们map
可以拿到我们对所有item
进行筛选的结果,true
或者false
。
接下来对原数组每一项进行返回对应下标的结果即可。
some
some
作为一个用来检测数组是否满足一些条件的函数存在,同样是可以用作遍历的
函数签名同forEach
,有区别的是当任一callback
返回值匹配为true
则会直接返回true
,如果所有的callback
匹配均为false
,则返回false
我们要判断数组中是否有元素等于2
:
[1, 2, 3].some(item => item === 2)
// > true
然后我们将它改为Promise
[1, 2, 3].some(async item => item === 2)
// > true
这个函数依然会返回true
,但是却不是我们想要的,因为这个是async
返回的Promise
对象被认定为true
。
所以,我们要进行如下处理:
Array.prototype.someSync = async function (callback, thisArg) {
for (let [index, item] of Object.entries(this)) {
if (await callback(item, index, this)) return true
}
return false
}
await [1, 2, 3].someSync(async item => item === 2)
// > true
因为some
在匹配到第一个true
之后就会终止遍历,所以我们在这里边使用forEach
的话是在性能上的一种浪费。
同样是利用了await
会忽略普通表达式的优势,在内部使用for-of
来实现我们的需求
every
以及我们最后的一个every
函数签名同样与forEach
一样,
但是callback
的处理还是有一些区别的:
其实换一种角度考虑,every
就是一个反向的some
some
会在获取到第一个true
时终止
而every
会在获取到第一个false
时终止,如果所有元素均为true
,则返回true
我们要判定数组中元素是否全部大于3
[1, 2, 3].every(item => item > 3)
// > false
很显然,一个都没有匹配到的,而且回调函数在执行到第一次时就已经终止了,不会继续执行下去。
我们改为Promise
版本:
[1, 2, 3].every(async => item > 3)
// > true
这个必然是true
,因为我们判断的是Promise
对象
所以我们拿上边的someSync
实现稍微修改一下:
Array.prototype.everySync = async function (callback, thisArg) {
for (let [index, item] of Object.entries(this)) {
if (!await callback(item, index, this)) return false
}
return true
}
await [1, 2, 3].everySync(async item => item === 2)
// > false
当匹配到任意一个false
时,直接返回false
,终止遍历。
javascript 遍历的更多相关文章
- JavaScript遍历table的行和列
来源:http://blog.csdn.net/bobwu/article/details/7497412 <HTML> <head> <SCRIPT LANGUAGE= ...
- JavaScript遍历对象-总结一
原生JavaScript 遍历 1.for 循环遍历 let array1 = ['a','b','c']; for (let i = 0;i < array1.length;i++){ con ...
- JavaScript遍历table
JavaScript遍历table 1.说明 遍历表格中的某行某列,并打印其值 2.实现源码 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML ...
- 如何利用JavaScript遍历JSON数组
1.设计源码 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www. ...
- javaScript遍历对象、数组总结(转载)
javaScript遍历对象.数组总结 转载来源 https://www.cnblogs.com/chenyablog/p/6477866.html 在日常工作过程中,我们对于javaScript遍 ...
- javaScript遍历对象、数组总结
javaScript遍历对象总结 1.使用Object.keys()遍历 返回一个数组,包括对象自身的(不含继承的)所有可枚举属性(不含Symbol属性). var obj = {'0':'a ...
- JavaScript遍历IP段内所有IP
思路:将两个IP转换为数字进行比较,小的那个慢慢加一,直到变成大的那个IP所转换的数字,将这其中的数字再转换为IP地址即为IP段内所有的IP. //IP转数字 function ip2int(ip) ...
- JavaScript 遍历对象、数组总结
在日常工作过程中,我们对于javaScript遍历对象.数组的操作是十分的频繁的,今天抽空把经常用到的方法小结一下,方便今后参考使用! javaScript遍历对象总结 1.使用Objec ...
- javascript遍历控件(实例详解)
js遍历页面控件, 代码如下 复制代码 var inputArr = document.forms[0]; for( var i = 0; i < inputArr.length; i++ ...
- JavaScript遍历方式详解
为了方便例子讲解,现有数组和json对象如下: var demoArr = ['Javascript', 'Gulp', 'CSS3', 'Grunt', 'jQuery', 'angular']; ...
随机推荐
- Tempdb--Snapshot
The insert operation does not cause a row version to be generated because there is really no prvious ...
- 启动hive命令时指定参数或自定义参数
启动hive命令时指定参数或自定义参数 在hive启动命令中指定一个参数 hive --hiveconf hive.job.submit.username=fuxin.zhao -e "se ...
- 浏览器兼容性随手记:Javascript
1.event IE9以下不支持直接获取event对象,所以需要写兼容: var event = event?event:window.event; IE8以下不支持event.target,但是可以 ...
- C#——做一个简单代理IP池
一.缘由. 抓取数据时,有一些网站 设置了一些反爬虫设置,进而将自己本地 IP 地址拉入系统黑名单.从而达到禁止本地 IP 访问数据的请求. 二.思路. 根据其他 代理 IP 网站,进行一个免费的代理 ...
- E - More is better (并查集)
点击打开链接 Mr Wang wants some boys to help him with a project. Because the project is rather complex, th ...
- 【OCP-12c】CUUG 071题库考试原题及答案解析(17)
17.(7-11) choose twoView the Exhibit and examine the structure of ORDER_ITEMS and ORDERS tables.You ...
- 配置DNS服务器转发器
1.(windows server 2008 r2)控制面板->管理工具->服务器管理器->DNS->服务器图标->属性 2.转发器->编辑 3.填写DNS并点击确 ...
- 脚本:定时释放 Linux/CentOS 缓存【转载自:杭州山不高】
定时释放Linux/CentOS缓存的脚本(yl_dropcaches)如下: #!/bin/bash used=`free -m | awk 'NR==2' | awk '{print $3}'` ...
- CTF中密码学一些基础
本文作者:i春秋签约作家MAX. 凯撒密码作为一种最为古老的对称加密体制,在古罗马的时候都已经很流行,他的基本思想是:通过把字母移动一定的位数来实现加密和解密. 给大家先找两道题,来一起探讨基础密码学 ...
- 浅析Postgres中的并发控制(Concurrency Control)与事务特性(下)
上文我们讨论了PostgreSQL的MVCC相关的基础知识以及实现机制.关于PostgreSQL中的MVCC,我们只讲了元组可见性的问题,还剩下两个问题没讲.一个是"Lost Update& ...