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

作者:laozhang 时间:2024-05-29 22:11:56 

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

来源:https://segmentfault.com/a/1190000012738395

标签:JavaScript,异步
0
投稿

猜你喜欢

  • Javascript连接Access数据库完整实例

    2024-01-15 22:16:58
  • Golang 如何判断数组某个元素是否存在 (isset)

    2024-02-04 17:11:46
  • Python 格式化打印json数据方法(展开状态)

    2023-10-07 05:29:50
  • 在Windows的Apache服务器上配置对PHP和CGI的支持

    2023-10-20 22:12:17
  • 简单代码实现可输入的下拉框功能(select)

    2008-10-20 19:52:00
  • IDEA最新激活码永久激活教程附激活失败原因汇总

    2023-12-13 11:57:29
  • python使用插值法画出平滑曲线

    2021-07-14 03:59:02
  • MySQL的一些常用的SQL语句整理

    2024-01-19 06:38:40
  • vue3.0如何使用computed来获取vuex里数据

    2024-04-28 09:24:20
  • Python 安装第三方库 pip install 安装慢安装不上的解决办法

    2023-02-23 13:43:45
  • 使用Python中tkinter库简单gui界面制作及打包成exe的操作方法(二)

    2023-04-25 19:32:40
  • Python全栈之线程详解

    2021-05-21 17:44:21
  • 首页访问感受提升三步曲

    2007-12-13 20:36:00
  • MySql安装与配置方法(MySQL添加用户、删除用户与授权)

    2024-01-25 07:25:01
  • Javascript中的基本类型和引用类型概述说明

    2024-04-18 09:37:04
  • 利用Python找回微信撤回信息

    2022-11-21 22:34:03
  • Python mutiprocessing多线程池pool操作示例

    2022-02-11 14:19:46
  • pytorch下使用LSTM神经网络写诗实例

    2022-03-12 13:21:43
  • Python 如何实现访问者模式

    2021-08-10 20:49:53
  • 使用SqlBulkCopy时应注意Sqlserver表中使用缺省值的列

    2012-07-11 15:34:35
  • asp之家 网络编程 m.aspxhome.com