JavaScript体验异步更好的解决办法


Posted in Javascript onJanuary 08, 2018

一、异步解决方案的进化史
JavaScript的异步操作一直是个麻烦事,所以不断有人提出它的各种解决方案。可以追溯到最早的回调函数(ajax老朋友),到Promise(不算新的朋友),再到ES6的Generator(强劲的朋友)。
几年前我们可能用过一个比较著名的Async.js,但是它没有摆脱回调函数,并且错误处理也是按照“回调函数的第一个参数用来传递错误”这样一个约定。而众所周知的回调地狱仍然是一个比较突出的问题,直到Generator改变了这种异步风格。
但是ES7的async await的出现(碉堡的新朋友),我们可以轻松写出同步风格的代码同时又拥有异步机制,可以说是目前最简单,最优雅,最佳的解决方案了。

二、async await语法
async await语法比较简单,可以认为是Generator的语法糖,比起星号和yield更具有语义化。下面一个简单的例子表示1秒之后输出hello world:

function timeout(ms) {
 return new Promise((resolve) => {
  setTimeout(resolve, ms);
 });
}
async function asyncPrint(value, ms) {
 await timeout(ms);
 console.log(value)
}
asyncPrint('hello world', 1000);

await只能用在async函数中,如果用在普通函数就会报错

await后面跟的是一个Promise对象(当然其它值也可以,但是会包装成一个立即resolve的Promise,也就没有意义了)

await会等待Promise的结果返回再继续执行

await等待的虽然是Promise对象,但是不必写.then(),直接可以得到返回值,将上面的代码微调,发现返回值result也是可以输出hello world:

function timeout(ms) {
 return new Promise((resolve) => {
  setTimeout(_ => {resolve('hello world')}, ms);
 });
}
async function asyncPrint(ms) {
 let result = await timeout(ms);
 console.log(result)
}
asyncPrint(1000);

三、async await错误处理

前面说了await等待的虽然是Promise对象,但是不必写.then(),所以其实也不用写.catch()了,直接用try catch就能捕捉错误,这样可以避免错误处理代码非常冗余和笨重,还是将上面的例子微调:

function timeout(ms) {
 return new Promise((resolve, reject) => {
  setTimeout(_ => {reject('error')}, ms);//reject模拟出错,返回error
 });
}
async function asyncPrint(ms) {
 try {
   console.log('start');
   await timeout(ms);//这里返回了错误
   console.log('end');//所以这句代码不会被执行了
 } catch(err) {
   console.log(err); //这里捕捉到错误error
 }
}
asyncPrint(1000);

如果有多个await,可以一起放在try catch中:

async function main() {
 try {
  const async1 = await firstAsync();
  const async2 = await secondAsync();
  const async3 = await thirdAsync();
 }
 catch (err) {
  console.error(err);
 }
}

四、async await注意点

1). 前面已经说过,await命令后面的Promise对象,运行结果很可能是reject或逻辑报错,所以最好把await放在try catch代码块中。

2). 多个await命令的异步操作,如果不存在依赖关系,让它们同时触发。

const async1 = await firstAsync();
const async2 = await secondAsync();

上面代码中,async1和async2如果是两个独立的异步操作,这样写会比较耗时,因为只有firstAsync完成以后,才会执行secondAsync,完全可以用Promise.all优雅地处理:

let [async1, async2] = await Promise.all([firstAsync(), secondAsync()]);

3). await只能用在async函数之中,如果用在普通函数就会报错:

async function main() {
 let docs = [{}, {}, {}];
 //报错 await is only valid in async function
 docs.forEach(function (doc) {
  await post(doc);
  console.log('main');
 });
}
function post(){
 return new Promise((resolve) => {
  setTimeout(resolve, 1000);
 });
}

在forEach内部方法加上async就可以了:

async function main() {
 let docs = [{}, {}, {}];
 docs.forEach(async function (doc) {
  await post(doc);
  console.log('main');
 });
}
function post(){
 return new Promise((resolve) => {
  setTimeout(resolve, 1000);
 });
}

但是你会发现3个main是同时输出的,这就说明post是并发执行的,而不是继发执行,改成for就可以解决问题,3个main是分别相隔1秒输出:

async function main() {
 let docs = [{}, {}, {}];
 for (let doc of docs) {
  await post(doc);
  console.log('main');
 }
}
function post(){
 return new Promise((resolve) => {
  setTimeout(resolve, 1000);
 });
}

总之,用了async await之后整个人神清气爽,可以用非常简洁和优雅的代码实现各种花式异步操作,并且在业务逻辑复杂的情况下可以不用陷入回调地狱中。不敢说这一定是终极的解决方案,但确实是目前最优雅的解决方案!

Javascript 相关文章推荐
javascript同步Import,同步调用外部js的方法
Jul 08 Javascript
js 新浪的一个图片播放图片轮换效果代码
Jul 15 Javascript
jquery实现图片滚动效果的简单实例
Nov 23 Javascript
从零学jquery之如何使用回调函数
May 16 Javascript
javascript实现网页屏蔽Backspace事件,输入框不屏蔽
Jul 21 Javascript
js立即执行函数: (function ( ){})( ) 与 (function ( ){}( )) 有什么区别?
Nov 18 Javascript
jQuery判断浏览器并动态调整select宽度的方法
Mar 02 Javascript
JS实现图片剪裁并预览效果
Aug 12 Javascript
详解用node编写自己的cli工具
May 23 Javascript
bootstrap表格内容过长时用省略号表示的解决方法
Nov 21 Javascript
详解Webpack + ES6 最新环境搭建与配置
Jun 04 Javascript
学习LayUI时自研的表单参数校验框架案例分析
Jul 29 Javascript
探索Vue高阶组件的使用
Jan 08 #Javascript
Vue入门之数据绑定(小结)
Jan 08 #Javascript
浅谈Vue数据绑定的原理
Jan 08 #Javascript
让网站自动生成章节目录索引的多个js代码
Jan 07 #Javascript
JavaScript学习总结(一) ECMAScript、BOM、DOM(核心、浏览器对象模型与文档对象模型)
Jan 07 #Javascript
Bootstrap treeview实现动态加载数据并添加快捷搜索功能
Jan 07 #Javascript
vue实现登录后页面跳转到之前页面
Jan 07 #Javascript
You might like
php查询ip所在地的方法
2014/12/05 PHP
百万级别知乎用户数据抓取与分析之PHP开发
2015/09/28 PHP
PHP实现的简单组词算法示例
2018/04/10 PHP
laravel Task Scheduling(任务调度)在windows下的使用详解
2019/10/22 PHP
基于jquery的一个浮动框(扩展性比较好 )
2010/08/27 Javascript
js实现一个省市区三级联动选择框代码分享
2013/03/06 Javascript
js弹出对话框方式小结
2015/11/17 Javascript
javascript针对不确定函数的执行方法
2015/12/16 Javascript
Bootstrap表单Form全面解析
2016/06/13 Javascript
浅谈JavaScript中的分支结构
2016/07/01 Javascript
JS实现拖动滚动条评分的效果代码分享
2016/09/29 Javascript
自己封装的一个简单的倒计时功能实例
2016/11/23 Javascript
完美实现js拖拽效果 return false用法详解
2017/07/28 Javascript
vue router使用query和params传参的使用和区别
2017/11/13 Javascript
vue实现导航栏效果(选中状态刷新不消失)
2017/12/13 Javascript
layui动态表头的实现代码
2019/08/22 Javascript
js实现蒙版效果
2020/01/11 Javascript
在微信小程序中渲染HTML内容3种解决方案及分析与问题解决
2020/01/12 Javascript
vue组件开发之tab切换组件使用详解
2020/08/21 Javascript
[05:34]2014DOTA2国际邀请赛中国区预选赛精彩TOPPLAY第二弹
2014/06/25 DOTA
[00:03]DOTA2新版本PA至宝展示
2014/11/19 DOTA
[01:43]深扒TI7聊天轮盘语音出处4
2017/05/11 DOTA
Python切片用法实例教程
2014/09/08 Python
python获取指定路径下所有指定后缀文件的方法
2015/05/26 Python
Python实现读取并保存文件的类
2017/05/11 Python
Python中%是什么意思?python中百分号如何使用?
2018/03/20 Python
教你使用python实现微信每天给女朋友说晚安
2018/03/23 Python
Python 实现选择排序的算法步骤
2018/04/22 Python
通过实例简单了解python yield使用方法
2020/08/06 Python
CSS3实现网站商品展示效果图
2020/01/18 HTML / CSS
Html5移动端获奖无缝滚动动画实现示例
2018/06/25 HTML / CSS
Rentalcars.com中国:世界上最大的在线汽车租赁服务
2019/08/22 全球购物
电子商务专业学生的学习自我评价
2013/10/27 职场文书
周鸿祎:教你写创业计划书
2013/12/30 职场文书
Redis调用Lua脚本及使用场景快速掌握
2022/03/16 Redis
Docker与K8s关系介绍不会Docker也可以使用K8s
2022/06/25 Servers