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 相关文章推荐
用JQuery调用Session的实现代码
Oct 29 Javascript
Validform+layer实现漂亮的表单验证特效
Jan 17 Javascript
Vue.js第四天学习笔记(组件)
Dec 02 Javascript
javascript简写常用的12个技巧(可以大大减少你的js代码量)
Mar 28 Javascript
JavaScript实现的DOM绘制柱状图效果示例
Aug 08 Javascript
浅谈Three.js截图并下载的大坑
Nov 01 Javascript
微信小程序服务器日期格式化问题
Jan 07 Javascript
javascript实现简单搜索功能
Mar 26 Javascript
vue使用map代替Aarry数组循环遍历的方法
Apr 30 Javascript
JavaScript异步操作的几种常见处理方法实例总结
May 11 Javascript
jQuery+ThinkPHP实现图片上传
Jul 23 jQuery
vue+vant 上传图片需要注意的地方
Jan 03 Vue.js
探索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的文件操作与算法实现的面试题示例
2015/08/10 PHP
详解PHP的Yii框架中日志的相关配置及使用
2015/12/08 PHP
JavaScript自定义方法实现trim()、Ltrim()、Rtrim()的功能
2013/11/03 Javascript
ExtJs中gridpanel分组后组名排序实例代码
2013/12/02 Javascript
JavaScript中的object转换成number或string规则介绍
2014/12/31 Javascript
jQuery 处理页面的事件详解
2015/01/20 Javascript
理解JavaScript中worker事件api
2015/12/25 Javascript
你不知道的高性能JAVASCRIPT
2016/01/18 Javascript
JavaScript function函数种类详解
2016/02/22 Javascript
JQuery解析XML的方法小结
2016/04/02 Javascript
Jq通过td获取同行其它列td的方法
2016/10/05 Javascript
微信开发之调起摄像头、本地展示图片、上传下载图片实例
2016/12/08 Javascript
js前端实现图片懒加载(lazyload)的两种方式
2017/04/24 Javascript
JS实现仿UC浏览器前进后退效果的实例代码
2017/07/17 Javascript
JS使用正则表达式找出最长连续子串长度
2017/10/26 Javascript
浅谈mint-ui loadmore组件注意的问题
2017/11/08 Javascript
除Console.log()外更多的Javascript调试命令
2018/01/24 Javascript
jQuery实现当拉动滚动条到底部加载数据的方法分析
2019/01/24 jQuery
微信小程序五子棋游戏的棋盘,重置,对弈实现方法【附demo源码下载】
2019/02/20 Javascript
浅谈小程序 setData学问多
2019/02/20 Javascript
JS如何实现在弹出窗口中加载页面
2020/12/03 Javascript
Python中subprocess模块用法实例详解
2015/05/20 Python
Python中列表、字典、元组数据结构的简单学习笔记
2016/03/20 Python
对Python的多进程锁的使用方法详解
2019/02/18 Python
Python3利用print输出带颜色的彩色字体示例代码
2019/04/08 Python
PyQt5 窗口切换与自定义对话框的实例
2019/06/20 Python
python 绘制拟合曲线并加指定点标识的实现
2019/07/10 Python
python批量修改xml属性的实现方式
2020/03/05 Python
如何利用python 读取配置文件
2021/01/06 Python
HTML5+Canvas+CSS3实现齐天大圣孙悟空腾云驾雾效果
2016/04/26 HTML / CSS
基层干部十八大感言
2014/01/19 职场文书
2014年巴西世界杯口号
2014/06/05 职场文书
中学生旷课检讨书500字
2014/10/29 职场文书
2014年妇幼卫生工作总结
2014/12/09 职场文书
上帝也疯狂观后感
2015/06/09 职场文书
2015初中政教处工作总结
2015/07/21 职场文书