JavaScript 的 Async\/Await 完胜 Promise 的六
参考:http://www.10tiao.com/html/558/201705/2650964601/1.html
Node 现在从版本 7.6 开始就支持 async/await 了。
简介:
Async/await 是一种编写异步代码的新方法。之前异步代码的方案是回调和 promise。
Async/await 实际上是建立在 promise 的基础上。它不能与普通回调或者 node 回调一起用。
Async/await 像 promise 一样,也是非阻塞的。
Async/await 让异步代码看起来、表现起来更像同步代码。这正是其威力所在。
语法:
假设函数 ajax 返回一个promise,而该promise的完成值是一些JSON对象。我们只想调用它,并输出该JSON,然后返回"done"。
如下是用 ajax 实现的代码:
var ajax=new Promise(function(resolve,reject){
$.ajax({
type:"post",
url:"list.php",
success:function(result){
/*
result = {
flag: true,
msg: '',
data: []
}
*/
if(result.flag){
resolve(data.data)//在异步操作成功时调用
}else{
reject(data.msg);//在异步操作失败时调用
}
}
});
});
如下是用 promise 实现的代码:
const makeRequest = () => ajax.then(data => {
console.log(data);
return "done";
})
makeRequest();
如下是用 async/await 实现的代码:
const makeRequest = async () => {
console.log(await ajax);
return 'done';
}
区别:
函数前面有一个关键字
async。await关键字只用在用async定义的函数内。所有async函数都会隐式返回一个 promise,而 promise 的完成值将是函数的返回值(本例中是"done")。不能在代码的顶层用
await,因为这样就不是在async函数内。
// await makeRequest() 这段代码在顶层不能执行
// 而下面这段代码可以执行
makeRequest().then((result) => {
// do something
})
3.await ajax 意味着 console.log 调用会一直等待,直到 ajax promise 完成并打印出它的值。
async/await 优点:
1. 简洁干净
节省了不少代码。不必写 .then,创建一个匿名函数来处理响应,或者给不需要用的变量一个名称 data。还避免了代码嵌套。这些小小的优势会快速累积起来,在后面的代码中会变得更明显。
2. 错误处理
Async/await 会最终让用同样的结构( try/catch)处理同步和异步代码变成可能。在下面使用 promise 的示例中,如果 JSON.parse 失败的话,try/catch 就不会处理,因为它是发生在一个 prmoise 中。需要在 promise 上调用 .catch,并且重复错误处理代码。这种错误处理代码会比可用于生产的代码中的 console.log 更复杂。
const makeRequest = () => {
try {
ajax.then(result => {
console.log(result);
})
} catch(err) {
// statements
console.log(err);
}
}
用 async/await 实现的代码。现在 catch 块会处理解析错误。
const makeRequest = async () => {
try {
// 这个会解析失败
const data = JSON.parse(await ajax);
console.log(data)
} catch (err) {
console.log(err);
}
}
3. 条件句
假设想做像下面的代码一样的事情,获取一些数据,并决定是否应该返回该数据,或者根据数据中的某些值获取更多的细节。
const makeRequest = () => {
return ajax.then(data => {
if (data.flag) {
return makeAnotherRequest(data).then(moreData => {
console.log(moreData);
return moreData;
})
} else {
console.log(data);
}
})
}
这些代码看着就让人头疼。它只需将最终结果传播到主 promise,却很容易让我们迷失在嵌套、大括号和返回语句中。
把这个示例用async / await 重写,就变得更易于阅读。
const makeRequest = async () => {
const data = await ajax;
if (data.flag) {
const moreData = await makeAnotherRequest(data);
console.log(moreData);
return moreData;
} else {
console.log(data);
}
}
4. 中间值
情景:调用 promise1,然后用它的返回值来调用promise2,然后使用这两个 promise 的结果来调用 promise3。你的代码很可能看起来像这样:
const makeRequest = () => {
return pomise1().then(value1 => {
return pomise2().then(value2 => {
return pomise3(value1, value2);
})
})
}
如果 promise3 不需要 value1,那么很容易就可以把 promise 嵌套变扁平一点。那么可能就会像下面这样,在一个 Promise.all中包含值 1 和 2,并避免更深层次的嵌套:
const makeRequest = () => {
return pomise1().then(value1 => {
return Pomise.all([
value1,
promise2(value1)
]).then(([value1, value2]) => {
return pomise3(value1, value2);
})
})
}
这种方法为了可读性而牺牲了语义。除了为了避免 promise 嵌套,没有理由将 value1和value2并入一个数组。
不过用 async/await 的话,同样的逻辑就变得超级简单直观了。
const makeRequest = async () => {
const value1 = await promise1();
const value2 = await promise2();
return promise3(value1, value2);
}
5. 错误栈
假如有一段链式调用多个 promise 的代码,在链的某个地方抛出一个错误。
// 错误栈
const makeRequest = () => {
return callAPromise().then(() => callAPromise())
.then(() => callAPromise())
.then(() => callAPromise())
.then(() => callAPromise())
.then(() => {
throw new Error('opps';)
})
} makeRequest().catch(err => {
console.log(err);
})
从 promise 链返回的错误栈没有发现错误发生在哪里的线索。更糟糕的是,这是误导的;它包含的唯一的函数名是callAPromise,它完全与此错误无关(不过文件和行号仍然有用)。
但是,来自async / await的错误栈会指向包含错误的函数:
const makeRequest = async() => {
await callAPromise();
await callAPromise();
await callAPromise();
await callAPromise();
await callAPromise();
throw new Error('oops');
}
makeRequest().catch(err => {
console.log(err);
})
当在本地环境中开发并在编辑器中打开文件时,这不是啥大事,但是当想搞清楚来自生产服务器的错误日志时,就相当有用了。在这种情况下,知道错误发生在makeRequest中比知道错误来自一个又一个的 then 要好。
6. 调试
最后但是同样重要的是,在使用 async/await 时,一个杀手级优势是调试更容易。调试 promise 一直是如此痛苦,有两个原因:
没法在返回表达式(无函数体)的箭头函数中设置断点。
// 错误栈
const makeRequest = () => {
return callAPromise().then(() => callAPromise())
.then(() => callAPromise())
.then(() => callAPromise())
.then(() => callAPromise())
.then(() => {
throw new Error('opps';)
})
}
试着在此处设置断点
2.如果在.then块中设置断点,并使用像单步调试这类调试快捷方式,调试器不会移动到后面的 .then ,因为它只单步调试同步代码。
有了 async/await,我们就不再需要那么多箭头函数,您可以像正常的同步调用一样单步调试 await 调用。
const makeRequest = async() => {
await callAPromise();
await callAPromise();
await callAPromise();
await callAPromise();
await callAPromise();
throw new Error('oops');
}
总结
Async/await 是过去几年中添加到 JavaScript 中的最具革命性的功能之一。它让我们意识到 promise 的语法有多混乱,并提供了直观的替代。
JavaScript 的 Async\/Await 完胜 Promise 的六的更多相关文章
- 6个Async/Await完胜Promise的原因
友情提醒:NodeJS自从7.6版开始已经内置了对async/await的支持.如果你还没用过该特性,那么接下来我会给出一系列的原因解释为何你应该立即开始使用它并且会结合示例代码说明. async/a ...
- JavaScript异步编程——Async/Await vs Promise
兼容性 提醒一下各位,Node 现在从版本 7.6 开始就支持 async/await 了.而就在前几天,Node 8已经正式发布了,你可以放心地使用它. 如果你还没有试过它,这里有一堆带有示例的理由 ...
- Async/Await替代Promise的6个理由
译者按: Node.js的异步编程方式有效提高了应用性能:然而回调地狱却让人望而生畏,Promise让我们告别回调函数,写出更优雅的异步代码:在实践过程中,却发现Promise并不完美:技术进步是无止 ...
- 8张图让你一步步看清 async/await 和 promise 的执行顺序
摘要: 面试必问 原文:8张图帮你一步步看清 async/await 和 promise 的执行顺序 作者:ziwei3749 Fundebug经授权转载,版权归原作者所有. 为什么写这篇文章? 说实 ...
- 8 张图帮你一步步看清 async/await 和 promise 的执行顺序(转)
https://mp.weixin.qq.com/s?__biz=MzAxODE2MjM1MA==&mid=2651555491&idx=1&sn=73779f84c289d9 ...
- [转] Async/Await替代Promise的6个理由
Node.js 7.6已经支持async/await了,如果你还没有试过,这篇博客将告诉你为什么要用它. Async/Await简介 对于从未听说过async/await的朋友,下面是简介: asyn ...
- 【转】6 Reasons Why JavaScript’s Async/Await Blows Promises Away (Tutorial)
原文:https://hackernoon.com/6-reasons-why-javascripts-async-await-blows-promises-away-tutorial-c7ec105 ...
- [转] 理解 JavaScript 的 async/await
[From] https://segmentfault.com/a/1190000007535316 边城 2016年11月19日发布 随着 Node 7 的发布,越来越多的人开始研究据说是 ...
- 理解 JavaScript 的 async/await
随着 Node 7 的发布,越来越多的人开始研究据说是异步编程终级解决方案的 async/await.我第一次看到这组关键字并不是在 JavaScript 语言里,而是在 c# 5.0 的语法中.C# ...
随机推荐
- 844. Backspace String Compare
class Solution { public: bool backspaceCompare(string S, string T) { int szs=S.size(); int szt=T.siz ...
- PHP中两个冒号是什么意思
类中 静态方法和静态属性的引用方法 对类的静态属性和方法的直接引用,这种情况可以不需要实例化类而直接使用“::”调用
- HMM(隐马尔可夫模型)不断学习中
HMM(隐马尔可夫模型)是用来描述隐含未知参数的统计模型,举一个经典的例子:一个东京的朋友每天根据天气{下雨,天晴}决定当天的活动{公园散步,购物,清理房间}中的一种,我每天只能在twitter上看到 ...
- UOJ 117 欧拉回路(套圈法+欧拉回路路径输出+骚操作)
题目链接:http://uoj.ac/problem/117 题目大意: 解题思路:先判断度数: 若G为有向图,欧拉回路的点的出度等于入度. 若G为无向图,欧拉回路的点的度数位偶数. 然后判断连通性, ...
- 远程算数程序——版本v1.0
很少有需要背诵的程序,但是从这个程序开始,标记的都是必须背诵的. 远程算数程序概述 远程算数程序比较简单,分为服务器端和客户端,客户端发送欲计算的表达式给服务器端,服务端经过计算又返回结果给客户端.如 ...
- output.php
<?php /** * */ class Output { function __construct() { $this->_zlib_oc = @ini_get('zlib.output ...
- 关于QT应用在XP系统上兼容运行的问题
修改兼容XP: 1. 项目属性->配置属性->平台工具集:Visual Studio 2013 - Windows XP (v120_xp) 2. C/C++ 属性-> 代码生成-& ...
- struts2马士兵笔记
Struts2 学习笔记 目录 01 Struts2-Action 一. Struts作用: 二. 搭建Struts2的运行环境: 三. Namespa ...
- 将一个浮点数转换成人民币读法字符串(java)
public class Num2Rmb { private String[] hanArr = {"零" , "壹" , "贰&qu ...
- AngularJS $scope 继承性 作用 生命周期
一.基本概念 作用域是一个指向应用模型的对象,相当于MVVM中的ViewModel,能绑定数据(属性)和行为(方法),能监控表达式和传递事件,是实现双向绑定的基础,是应用在 HTML (视图) 和 J ...