ES6 提供的 Promise 方法和 ES7 提供的 Async/Await 语法糖都可以更好解决多层回调问题, 详细用法可参考:https://www.cnblogs.com/cckui/p/9915604.html,下面进一步介绍 promise 和 async/await 用法的异同。

首先定义以下三个异步函数:

function sleep3000() {
return new Promise(function (resolve, reject) {
setTimeout(function () {
console.log('sleep3000 执行完成');
resolve(new Date());
}, 3000);
})
}; function sleep2000() {
return new Promise(function (resolve, reject) {
setTimeout(function () {
console.log('sleep2000 执行完成');
resolve(new Date());
}, 2000);
})
};
function sleep4000() {
return new Promise(function (resolve, reject) {
setTimeout(function () {
console.log('sleep4000 执行完成');
resolve(new Date());
}, 4000);
})
};

1. 使用 promise 方法分别执行以上三个函数

console.log('Now:', new Date());

sleep3000().then((d) => {
console.log('3000:', d);
}) sleep2000().then((d) => {
console.log('2000:',d);
}) sleep4000().then((d) => {
console.log('2000:',d);
})

最终打印结果:

Now: 2019-04-08T02:29:38.388Z
sleep2000 执行完成
2000: 2019-04-08T02:29:40.398Z
sleep3000 执行完成
3000: 2019-04-08T02:29:41.398Z
sleep4000 执行完成
2000: 2019-04-08T02:29:42.394Z

2. 使用 promise.all() 方法

Promise 的 all() 方法提供了并行执行异步操作的能力,即在所有异步操作执行完后才执行回调, all()里面接收一个数组,数组中每个元素都返回一个 promise 对象。

all()里面的异步都执行完成后,才会把每个异步结果以数组的形式放入 then 中。

某些情况下,如果上面sleep3000、sleep200、sleep4000分别代表页面不同模块的加载时间,在页面刚打开时,需要当所有模块加载完成后,清除loading。此时可以作如下处理:

console.log('Now:', new Date());

Promise.all([
sleep2000(),
sleep3000(),
sleep4000()
]).then((res) => {
console.log(res)
})

最终打印结果:

Now: 2019-04-08T02:43:07.139Z
sleep2000 执行完成
sleep3000 执行完成
sleep4000 执行完成
[ 2019-04-08T02:43:09.144Z,
2019-04-08T02:43:10.145Z,
2019-04-08T02:43:11.144Z ]

3. 使用 promise.race() 方法

Promise 的 race() 方法 与 all () 相反,即只以执行最快的那个promise为准,一旦race()中的某个promise解决或拒绝,所有返回的 promise就会解决或拒绝。

console.log('Now:', new Date());

Promise.race([
sleep2000(),
sleep3000(),
sleep4000()
]).then((res) => {
console.log(res)
})

因为 sleep2000 最先执行完成,所以在2秒后,就会立即进入then。

最终打印结果:

Now: 2019-04-08T02:44:34.193Z
sleep2000 执行完成
2019-04-08T02:44:36.200Z
sleep3000 执行完成
sleep4000 执行完成

注意:2秒进入 then 后, sleep3000 和 sleep4000 并没有被停止, 仍会继续走完,只是不会处理resolve或reject。

4. async/await 处理以上三个函数

console.log('Now:', new Date());

async function getDate () {
let res1 = await sleep3000();
let res2 = await sleep2000();
let res3 = await sleep4000();
console.log('3000:', res1);
console.log('2000:', res2);
console.log('4000:', res3);
} getDate();

最终打印结果:

Now: 2019-04-08T02:58:23.990Z
sleep3000 执行完成
sleep2000 执行完成
sleep4000 执行完成
3000: 2019-04-08T02:58:26.996Z
2000: 2019-04-08T02:58:29.001Z
4000: 2019-04-08T02:58:33.004Z

通过打印时间,可见,每处理完前一个await,才会处理下一个await,上面所有处理完花费了9秒。

所以, 只有当处理后面await需要前一个await返回值时,才可以用以上方法。

5. async/await 中使用 promise.all 并行处理异步操作

当三个异步相互之间没有关系,需要同时发送时:

console.log('Now:', new Date());

async function getdate () {
let res1 = sleep3000();
let res2 = sleep2000();
let res3 = sleep4000();
let res = await Promise.all([res1, res2, res3]);
console.log(res);
} getdate();

最终打印结果:

sleep2000 执行完成
sleep3000 执行完成
sleep4000 执行完成
[ 2019-04-08T03:11:24.280Z,
2019-04-08T03:11:23.280Z,
2019-04-08T03:11:25.275Z ]

注意:

对比 2 和 5 的打印结果,可以看出,二者打印的返回数组里面的顺序是有区别的:

  • 只用promise.all()时,各个异步是无序的;
  • 在 async/await 中使用 promise.all(), 各个异步是有序的。

异步操作之 Promise 和 Async await 用法进阶的更多相关文章

  1. Promise 和async/await 的使用理解

    Promise 和async/await 的使用理解 1. new Promise时就会开始执行语句. new Promise(resolve => resolove('成功信息') )     ...

  2. 20分钟带你掌握JavaScript Promise和 Async/Await

    转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. 原文出处:https://www.freecodecamp.org/news/learn-promise-a ...

  3. Promise和async await详解

    本文转载自Promise和async await详解 Promise 状态 pending: 初始状态, 非 fulfilled 或 rejected. fulfilled: 成功的操作. rejec ...

  4. “setTimeout、Promise、Async/Await 的区别”题目解析和扩展

    解答这个题目之前,先回顾下JavaScript的事件循环(Event Loop). JavaScript的事件循环 事件循环(Event Loop):同步和异步任务分别进入不同的执行"场所& ...

  5. node.js异步控制流程 回调,事件,promise和async/await

    写这个问题是因为最近看到一些初学者用回调用的不亦乐乎,最后代码左调来又调去很不直观. 首先上结论:推荐使用async/await或者co/yield,其次是promise,再次是事件,回调不要使用. ...

  6. Promise, Generator, async/await的渐进理解

    作为前端开发者的伙伴们,肯定对Promise,Generator,async/await非常熟悉不过了.Promise绝对是烂记于心,而async/await却让使大伙们感觉到爽(原来异步可以这么简单 ...

  7. Promise及Async/Await

      一.为什么有Async/Await? 我们都知道已经有了Promise的解决方案了,为什么还要ES7提出新的Async/Await标准呢? 答案其实也显而易见:Promise虽然跳出了异步嵌套的怪 ...

  8. callback vs async.js vs promise vs async / await

    需求: A.依次读取 A|B|C 三个文件,如果有失败,则立即终止. B.同时读取 A|B|C 三个文件,如果有失败,则立即终止. 一.callback 需求A: let read = functio ...

  9. Callback, Promise和Async/Await的对比

    Callback, Promise和Async/Await的对比 Callback Hell getData1(function (data1) { console.log('我得到data1了') ...

随机推荐

  1. Solr5.5高级应用(基于tomcat9)

    一.配置solr 1.配置 注意:要是想放到其它路径下,可以修改此路径下的web.xml配置文件 修改内容如下: <!-- 将solrhome的绝对路径写入env-entry-value --& ...

  2. SQL语句计算距离今天生日还差几天

    转载于:http://www.w3dev.cn/article/20110125/sql-compute-birthdate-now-days.aspx SQL语句计算距离生日还差几天原理很简单,将要 ...

  3. 智能合约 solidity 开发的环境基本搭建

    以太坊Dapp开发快速入门 以太坊为开源社区,虽然设计东西都很优秀,但是组件十分的杂乱,因此下面首先简单介绍下以太坊的一些常用组件以及各种工具介绍 Geth Geth是由以太坊基金会提供的官方客户端软 ...

  4. c/c++ 模板与STL小例子系列<一 >自建Array数组

    c/c++ 模板与STL小例子系列 自建Array数组 自建的Array数组,提供如下对外接口 方法 功能描述 Array() 无参数构造方法,构造元素个数为模板参数个的数组 Array(int le ...

  5. Java循环语句怎么用?经典排序算法见真知

    Java中循环语句的使用,莫过于在排序算法中使用得最为经典. 排序算法非常的多,不过大体可以分为两种: 一种是比较排序,主要有:冒泡排序,选择排序,插入排序,归并排序,堆排序,快速排序等. 另一种是非 ...

  6. netsh winsock reset命令,作用是重置 Winsock 目录

    Win xp重置 编辑 要为 Windows XP 重置 Winsock,请按照下列步骤操作: 1.单击“开始”,运行中输入cmd. 2.然后输入命令 netsh winsock reset. 3.重 ...

  7. Maven中POM.XML详解

    转自https://blog.csdn.net/jariwsz/article/details/19554137 我们先看一个简单的例子: <project xmlns="http:/ ...

  8. linux历史命令查找快捷方式

      一.回到上次操作的目录# cd -进入上次访问目录 二.历史命令搜索操作快捷键:[Ctrl + r], [Ctrl + p], [Ctrl + n] 在终端中按捉 [Ctrl] 键的同时 [r] ...

  9. (转)Geoserver基础配图研究

    https://blog.csdn.net/zbcx_ZGIS/article/details/82216151 1面图层配图 1.1基本框架 这个是geoserver默认的polygon的代码,我们 ...

  10. js模块化规范—AMD规范

    AMD规范说明 AMD全称是:Asynchronous Module Definition(异步模块定义),github地址 是专门用于浏览器端, 模块的加载是异步的 AMD规范基本语法 定义暴露模块 ...