首页 > 语言 > JavaScript > 正文

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

2024-05-06 15:22:48
字体:
来源:转载
供稿:网友

一、异步解决方案的进化史
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注意点

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表

图片精选