译者按: 通过真实的代码示例感受Async/Await的力量。
为了保证可读性,本文采用意译而非直译。另外,本文版权归原作者所有,翻译仅用于学习。
既然Node.js 8已经LTS了,我想大家是时候试一试Async/Await特性了,真的很好用!它可以帮助我们用同步的方式写异步代码,极大地提高了代码的可读性。在过去的2年时间里,Promise给我们带来了不少便利,同时也让我们有一些失望。
这边博客,我将介绍一个真实的代码示例,它是一个REST API的controller。通过展示我们如何从Promise切换到async/await,你讲能够体会到Async/Await的神奇之处!
Promise示例
下面是我的工作项目中真实的Controller代码:
const BPromise = require('bluebird');
const { WrongCredentialsError, DBConnectionError, EmailError } = require('./../errors');
/** * Emulate an Express.js route call as an example */ loginController({}, { json: response => console.log(response) }, null)
function loginController (req, res, err) { const { email, password } = req;
let user;
BPromise.try(() => validateUserInput(req)) .then(() => fetchUserByEmail(email)) .then(fetchedUser => user = fetchedUser) .then(() => comparePasswords(req.password, user.password)) .then(() => markLoggedInTimestamp(user.userId)) .then(() => sendEmail(user.userId)) .then(() => generateJWT(user)) .then(token => res.json({ success: true, token })) .catch(WrongCredentialsError, () => res.json({ success: false, error: 'Invalid email and/or password' })) .catch(EmailError, DBConnectionError, () => res.json({ success: false, error: 'Unexpected error, please try again' })) .catch(() => res.json({ success: false })) }
/** * Validate input from Request * * @param {Object} input * @throws {WrongCredentialsError} * @returns {Void} */ function validateUserInput(input) { if (!input.email || !input.password) { throw new WrongCredentialsError(); } }
/** * Fetch a User from the DB by Email * * @throws WrongCredentialsError * @throws DBConnectionError * @returns {BPromise} */ function fetchUserByEmail(email) { const user = { userId: 'DUMMY_ID', email: 'konmpar@gmail.com', password: 'DUMMY_PASSWORD_HASH' } return new BPromise(resolve => resolve(user)); }
/** * Compare two password * * @param {String} inputPwd * @param {String} storedPwd * @throws {WrongCredentialsError} * @returns {Void} */ function comparePasswords(inputPwd, storedPwd) { if (hashPassword(inputPwd) !== storedPwd) { throw new WrongCredentialsError(); } }
/** * Hash password * * @param {String} password * @returns {String} */ function hashPassword(password) { return password; }
/** * Mark a user's logged in timestamp * * @param {String} userId * @throws DBConnectionError * @returns {BPromise} */ function markLoggedInTimestamp(userId) { return new BPromise(resolve => resolve()); }
/** * Send a follow up email * * @param {String} userId * @throws EmailError * @returns {BPromise} */ function sendEmail(userId) { return new BPromise(resolve => resolve()); }
/** * Generate a JWT token to send to the client * * @param {Object} user * @returns {BPromise<String>} */ function generateJWT(user) { const token = 'DUMMY_JWT_TOKEN';
return new BPromise(resolve => resolve(token)); }
|
一些值得注意的要点:
多余的外层变量
let user;
/* ... */ .then(fetchedUser => user = fetchedUser) /* ... */ .then(() => sendEmail(user.userId)) /* ... */
|
可知,user是一个全局变量,因为我需要在Promise链中使用它。如果不希望定义多余的外层变量,则需要在Promise链中的每一个函数中都返回user变量,这样做显然更加糟糕。
烦人的Promise链
/* ... */ BPromise.try(() => validateUserInput(req)) /* ... */
|
一个Promise链必须从Promise开始,但是validateUserInput函数并没有返回Promise,这时需要使用Bluebird。我也知道这样写比较奇怪…
讨厌的Bluebird
我在很多地方都使用了Bluebird,如果不用它的话,代码会更加臃肿。所谓DRY,即Don’t repeat yourself,我们可以使用Bluebird去尽量简化代码。但是,Bluebird是一个第三方依赖,如果出问题了怎么办?去掉Bluebird应该更好!
JavaScript太灵(gui)活(yi)了,出了BUG你也不知道,不妨接入Fundebug线上实时监控。
Async/Await示例
当我放弃Promise,使用Async/Await之后,代码是这样的:
const { WrongCredentialsError, DBConnectionError, EmailError } = require('./../errors');
/** * Emulate an Express.js route call as an example */ loginController({}, { json: response => console.log(response) }, null)
/** * * @param {Object} req * @param {Object} res * @param {Object} err * @returns {Void} */ async function loginController(req, res, err) { const { email, password } = req.email;
try { if (!email || !password) { throw new WrongCredentialsError(); }
const user = await fetchUserByEmail(email);
if (user.password !== hashPassword(req.password)) { throw new WrongCredentialsError(); }
await markLoggedInTimestamp(user.userId); await sendEmail(user.userId);
const token = await generateJWT(user);
res.json({ success: true, token });
} catch (err) { if (err instanceof WrongCredentialsError) { res.json({ success: false, error: 'Invalid email and/or password' }) } else if (err instanceof DBConnectionError || err instanceof EmailError) { res.json({ success: false, error: 'Unexpected error, please try again' }); } else { res.json({ success: false }) } } }
/** * Fetch a User from the DB by Email * * @throws WrongCredentialsError * @throws DBConnectionError * @returns {Promise} */ function fetchUserByEmail(email) { const user = { userId: 'DUMMY_ID', email: 'konmpar@gmail.com', password: 'DUMMY_PASSWORD_HASH' } return new Promise(resolve => resolve(user)); }
/** * Hash password * * @param {String} password * @returns {String} */ function hashPassword(password) { return password; }
/** * Mark a user's logged in timestamp * * @param {String} userId * @throws DBConnectionError * @returns {Promise} */ function markLoggedInTimestamp(userId) { return new Promise(resolve => resolve()); }
/** * Send a follow up email * * @param {String} userId * @throws EmailError * @returns {Promise} */ function sendEmail(userId) { return new Promise(resolve => resolve()); }
/** * Generate a JWT token to send to the client * * @param {Object} user * @returns {Promise<String>} */ function generateJWT(user) { const token = 'DUMMY_JWT_TOKEN';
return new Promise(resolve => resolve(token)); }
|
哈哈!!!
没有外层变量
现在,所有函数都在同一个作用域中调用,不再需要.then函数。因此,我们不再需要定义多余的全局变量,也不需要做多余的变量赋值。
没有多余的函数
Promise示例中的同步函数validateInput和comparePasswords的代码可以与异步函数写在一起,因此可以不再需要定义单独的函数,代码更少。
可读性更高
异步代码采用同步方式来写,同时减少了代码量,可读性大大提高。
不再需要Bluebird
原生的Promise可以替代Bluebird,且不再需要Bluebird的try方法了。
结论
作为程序员,我们应该努力完善代码。Async/Await可以带来很大好处,帮助我们写出可读性更高的代码。如果你坚持使用Promise,不妨看看如何在Promise链中共享变量?。
如果你对Async/Await感兴趣的话,可以看看这些博客:
版权声明:
转载时请注明作者Fundebug以及本文地址:
https://blog.fundebug.com/2018/01/31/a-real-async-await-example/
- Async/Await 学习与示例
参考:Async/await学习 es 7 提供了对 promise 对象的更好的操作,省去了很多丧心病狂的链式异步请求,promise 是回调地狱的福音,而 Async/Await 则是 promi ...
- [.NET] 利用 async & await 的异步编程
利用 async & await 的异步编程 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/5922573.html 目录 异步编程的简介 异 ...
- [.NET] 利用 async & await 进行异步 IO 操作
利用 async & await 进行异步 IO 操作 [博主]反骨仔 [出处]http://www.cnblogs.com/liqingwen/p/6082673.html 序 上次,博主 ...
- [C#] 走进异步编程的世界 - 开始接触 async/await
走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $&qu ...
- ASP.NET 中的 Async/Await 简介
本文转载自MSDN 作者:Stephen Cleary 原文地址:https://msdn.microsoft.com/en-us/magazine/dn802603.aspx 大多数有关 async ...
- EntityFramework 如何进行异步化(关键词:async·await·SaveChangesAsync·ToListAsync)
应用程序为什么要异步化?关于这个原因就不多说了,至于现有项目中代码异步化改进,可以参考:实际案例:在现有代码中通过async/await实现并行 这篇博文内容针对的是,EntityFramework ...
- 走进异步编程的世界 - 开始接触 async/await
[C#] 走进异步编程的世界 - 开始接触 async/await 走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async ...
- ASP.NET 上的 Async/Await 简介
原文链接 大多数有关 async/await 的在线资源假定您正在开发客户端应用程序,但在服务器上有 async 的位置吗?可以非常肯定地回答“有”.本文是对 ASP.NET 上异步请求的概念性概述, ...
- Python PEP 492 中文翻译——协程与async/await语法
原文标题:PEP 0492 -- Coroutines with async and await syntax 原文链接:https://www.python.org/dev/peps/pep-049 ...
随机推荐
- 【洛谷4172】 [WC2006]水管局长(LCT)
传送门 洛谷 BZOJ Solution 如果不需要动态的话,那就是一个裸的最小生成树上的最大边权对吧. 现在动态了的话,把这个过程反着来,就是加边对吧. 现在问题变成了怎么动态维护加边的最小生成树, ...
- 每天学点SpringCloud(七):路由器和过滤器-Zuul
为什么要使用Zuul 先来看一下下方这个图 假如现在我们具有四个微服务,分别是用户.订单.支付.催收微服务,它们的调用方式分别是使用http.restful.thrift.kafka.这个时候如果我们 ...
- TextView文字描边实现
TextView文字描边实现 需求描述 文字显示在图片的上面,图片的内容是不确定了,为了防止文字与图片的颜色相近导致用户看不到或者看不清文字的问题,所以显示文字描边,避免问题. 实现 实现思想 使用T ...
- PHP调用百度天气接口API
//百度天气接口API $location = "北京"; //地区 $ak = "5slgyqGDENN7Sy7pw29IUvrZ"; //秘钥,需要申请,百 ...
- while(true)应用之 实现自己的消息队列
早些时候,一直有个疑问,就是比如你从前端发一个操作之后,后台为什么能够及时处理你的东西呢?当然了,我说的不是,服务器为什么能够立即接收到你的请求之类高大上的东西.而是,假设你用异步去做一个事情,而后台 ...
- js-完整轮播图
js-完整轮播图 今天写一个完整的轮播图,首先它需要实现三个功能:1.鼠标放在小圆点上实现轮播.2.点击焦点按钮实现轮播.3.无缝自动轮播. 轮播图的原理: 一系列的大小相等的图片平铺,利用CSS布 ...
- (转)asyncio --- 异步 I/O
原文:https://docs.python.org/zh-cn/3/library/asyncio.html asyncio 是用来编写 并发 代码的库,使用 async/await 语法. asy ...
- android项目架构 -----Android 知识体系与常用第三方框架
好东西值得分享 ,这是网络上总结的一些开源的东西直接就拿过来了 .... Android通用流行框架大全 先把这张图放在这 ,先来谈一谈项目结构 .我喜欢将东西按模块来划分: 都知道module . ...
- 微信小程序注册60s倒计时功能 使用JS实现注册60s倒计时功能
微信小程序+WEB使用JS实现注册[60s]倒计时功能开发步骤: 1.wxml页面代码: <text>绑定手机</text> <form bindsubmit=" ...
- Java 基础:hashCode方法
Writer:BYSocket(泥沙砖瓦浆木匠) 微博:BYSocket 豆瓣:BYSocket 一.前言 泥瓦匠最近被项目搞的天昏地暗.发现有些要给自己一些目标,关于技术的目标: 专注很重要.专注J ...