JavaScript异步编程之async,await(翻译)

原文地址:https://javascript.info/async-await

有一种特殊的语法可以以更舒适的方式处理promises,称为“async / await”。在理解和使用起来非常简单。

目录

  • Async
  • Await
  • 错误处理
  • 总结
  • 参考

Async

在函数前面添加async意味着一件简单的事:函数总是返回一个promise。如果代码中包含return <non-promise>,JavaScript则会自动将其包装到返回的promiseresolved

async function f() {
    return 1
}

//使用
f().then(console.log)    //1
//等同于
f().then(res=>{
    console.log(res)    //1
})

上述代码等同于我们可以明确返回一个promise

async function f() {
    return Promise.resolve(1)
}

//使用
f().then(console.log)    //1

因此,async确保了函数会返回一个promise,并在其中包含非promise代码。很简单吧!但不仅如此,还有另一个关键字await,它只能在异步函数中运行,而且非常酷。

Await

基本语法

//仅适用于异步函数
let value = await promise;

关键词await可以让JavaScript等到promise结束后再返回它的值
这里有一个promise1秒之后才resolve()的例子

async function f() {
    let promise = new Promise((resolve, reject) => {
          setTimeout(() => resolve("done!"), 1000)
    })

    let result = await promise; // wait till the promise resolves (*)

    alert(result); // "done!"
}

f()

函数在(*)一行执行“暂停”,直到promise结算完毕后才继续执行,同时result变成了这个函数的返回值。所以上述代码会在1秒之后弹出“done!”

强调await字面量意思是让JavaScript等到promise结束,然后继续执行。这不会花费任何CPU资源,因为引擎可以同时执行其他任务:执行其他脚本、处理事件等。

它只是一种更优雅的语法来获得promise结果,而不是用promise.then,更容易阅读和编写。

⚠ 不能在非异步函数中使用await

如果我们尝试在非异步函数中使用await,那将是语法错误:

function f() {
  let promise = Promise.resolve(1);
  let result = await promise; // Syntax error
}

如果我们忘记在函数之前放置异步(async),我们会得到这样的错误。如上所述,await仅在异步函数内部工作。

⚠ 不能在顶级函数中使用await1

刚开始使用await的人往往会忘记这一点,但我们不能在顶级函数中写await,这不起作用。

// syntax error in top-level code
let response = await fetch('/article/promise-chaining/user.json');
let user = await response.json();

因此,我们需要为await的代码提供包装异步功能,就像上面的例子一样。

await接受那些具有可调用的then方法的对象

就像promise.then一样,await允许使用thenable对象(那些具有可调用的then方法)。这意味着,第三方对象可能不是promise对象,但只要可调用then方法,那么这足以与await一起使用。
例如,这里await接受new Thenable(1)

class Thenable {
  constructor(num) {
    this.num = num;
  }
  then(resolve, reject) {
    alert(resolve); // function() { native code }
    // resolve with this.num*2 after 1000ms
    setTimeout(() => resolve(this.num * 2), 1000); // (*)
  }
};

async function f() {
  // waits for 1 second, then result becomes 2
  let result = await new Thenable(1);
  alert(result);
}

f();

⚠ 异步方法

类方法也可以是异步的,只需在它之前放入async

class Waiter {
  async wait() {
    return await Promise.resolve(1);
  }
}

new Waiter()
  .wait()
  .then(alert); // 1

错误处理

如果promise解析正常,则await promise返回结果。但是在返回reject的情况下,就需要使用throw语句抛出错误。

async function f() {
  await Promise.reject(new Error("Whoops!"));
}

等同于

async function f() {
  throw new Error("Whoops!");
}

在实际情况中,promise可能需要一些时间才能返回reject。所以await会等待,然后抛出一个错误。 我们可以使用try..catch捕获该错误,与常规throw相同:

async function f() {

  try {
    let response = await fetch('http://no-such-url');
  } catch(err) {
    alert(err); // TypeError: failed to fetch
  }
}

f();

如果出现错误,控件将跳转到catch块。我们还可以包装多行:

async function f() {

  try {
    let response = await fetch('/no-user-here');
    let user = await response.json();
  } catch(err) {
    // catches errors both in fetch and response.json
    alert(err);
  }
}

f();

如果我们没有try..catch,则异步函数f()调用生成的promise将被拒绝。我们可以附加.catch来处理它:

async function f() {
  let response = await fetch('http://no-such-url');
}

// f() becomes a rejected promise
f().catch(alert); // TypeError: failed to fetch // (*)

async/awaitpromise.then/catch

当我们使用async/await时,我们很少需要.then,因为await处理等待事件。我们可以使用常规的try..catch而不是.catch。这通常(并非总是)更方便。 但是在代码的顶层,当我们在任何异步函数之外时,我们在语法上就无法使用await,因此通常的做法是添加.then/catch来处理最终结果或错误处理。 就像上面例子的(*)一行一样。

async/await 也很好地适用于 Promise.all

当我们需要等待多个promise时,我们可以将它们包装在Promise.all中然后使用await

// wait for the array of results
let results = await Promise.all([
  fetch(url1),
  fetch(url2),
  ...
]);

如果出现错误,它会像往常一样传播:在Promise.all中返回reject,然后我们可以使用try..catch捕获的异常。

总结

async关键字放在函数前的有两个效果:

1.使它始终返回一个promise
2.允许在其中使用await
await关键字放在promise之前,函数会在执行中“暂停”,直到promise结算完毕后才继续执行:

如果是错误,则会生成异常,就像throw error在该位置一样调用。
否则,它返回结果,因此我们可以将其指定成一个值。
它们共同提供了一个很好的框架来编写易于读写的异步代码。

随着async/await的发布,我们很少需要写promise.then/catch,但我们仍然不应该忘记,他们是基于promise,因为有时(例如,在函数最外面)我们必须使用这些方法。同时Promise.all可以等待许多任务也是一件好事。

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351

推荐阅读更多精彩内容