async和await

async作用

async声明function是一个异步函数,返回一个promise对象,可以使用 then 方法添加回调函数。async函数内部return语句返回的值,会成为then方法回调函数的参数。


async function test() {

  return 'test';

}

console.log(test); // [AsyncFunction: test] async函数是[`AsyncFunction`]构造函数的实例

console.log(test()); // Promise { 'test' }


// async返回的是一个promise对象

test().then(res=>{

  console.log(res); // test

})


// 如果async函数没有返回值 async函数返回一个undefined的promise对象

async function fn() {

  console.log('没有返回');

}

console.log(fn()); // Promise { undefined }


// 可以看到async函数返回值和Promise.resolve()一样,将返回值包装成promise对象,如果没有返回值就返回undefined的promise对象

await

await 操作符只能在异步函数 async function 内部使用。如果一个 Promise 被传递给一个 await 操作符,await 将等待 Promise 正常处理完成并返回其处理结果,也就是说它会阻塞后面的代码,等待 Promise 对象结果。如果等待的不是 Promise 对象,则返回该值本身。


async function test() {

  return new Promise((resolve)=>{

    setTimeout(() => {

        resolve('test 1000');

    }, 1000);

  })

}

function fn() {

  return 'fn';

}


async function next() {

    let res0 = await fn(),

        res1 = await test(),

        res2 = await fn();

    console.log(res0);

    console.log(res1);

    console.log(res2);

}

next(); // 1s 后才打印出结果 为什么呢 就是因为 res1在等待promise的结果 阻塞了后面代码。

错误处理

如果await后面的异步操作出错,那么等同于async函数返回的 Promise 对象被reject。


async function test() {

  await Promise.reject('错误了')

};


test().then(res=>{

  console.log('success',res);

},err=>{

  console.log('err ',err);

})

// err 错误了

防止出错的方法,也是将其放在try...catch代码块之中。


async function test() {

  try {

    await new Promise(function (resolve, reject) {

      throw new Error('错误了');

    });

  } catch(e) {

      console.log('err', e)

  }

  return await('成功了');

}

多个await命令后面的异步操作,如果不存在继发关系(即互不依赖),最好让它们同时触发。


let foo = await getFoo();

let bar = await getBar();

// 上面这样写法 getFoo完成以后,才会执行getBar


// 同时触发写法 ↓


// 写法一

let [foo, bar] = await Promise.all([getFoo(), getBar()]);


// 写法二

let fooPromise = getFoo();

let barPromise = getBar();

let foo = await fooPromise;

let bar = await barPromise;

async/await优点

async/await的优势在于处理由多个Promise组成的 then 链,在之前的Promise文章中提过用then处理回调地狱的问题,async/await相当于对promise的进一步优化。 假设一个业务,分多个步骤,且每个步骤都是异步的,而且依赖上个步骤的执行结果。


// 假设表单提交前要通过俩个校验接口


async function check(ms) { // 模仿异步

  return new Promise((resolve)=>{

    setTimeout(() => {

        resolve(`check ${ms}`);

    }, ms);

  })

}

function check1() {

  console.log('check1');

  return check(1000);

}

function check2() {

  console.log('check2');

  return check(2000);

}


// -------------promise------------

function submit() {

  console.log('submit');

  // 经过俩个校验 多级关联 promise传值嵌套较深

  check1().then(res1=>{

    check2(res1).then(res2=>{

      /*

        * 提交请求

        */

    })

  })

}

submit();


// -------------async/await-----------

async function asyncAwaitSubmit() {

    let res1 = await check1(),

        res2 = await check2(res1);

        console.log(res1, res2);

        /*

        * 提交请求

        */

}



原理

async 函数的实现原理,就是将 Generator 函数和自动执行器,包装在一个函数里。


async function fn(args) {

  // ...

}


// 等同于


function fn(args) {

  return spawn(function* () {

    // ...

  });

}

/*

* Generator 函数就是一个封装的异步任务,或者说是异步任务的容器。

* 异步操作需要暂停的地方,都用 yield 语句注明

* 调用 Generator 函数,返回的是指针对象(这是它和普通函数的不同之处),。调用指针对象的 next 方法,会移动内部指针。

* next 方法的作用是分阶段执行 Generator 函数。每次调用 next 方法,会返回一个对象,表示当前阶段的信息( value 属性和 done 属性)。value 属性是 yield 语句后面表达式的值,表示当前阶段的值;done 属性是一个布尔值,表示 Generator 函数是否执行完毕,即是否还有下一个阶段。

*/


// 了解generator的用法

function* Generator() {

  yield '1';

  yield  Promise.resolve(2);

  return 'ending';

}

var gen = Generator(); // 返回指针对象 Object [Generator] {}


let res1 = gen.next();

console.log(res1); // 返回当前阶段的值 { value: '1', done: false }


let res2 = gen.next();

console.log(res2); // 返回当前阶段的值 { value: Promise { 2 }, done: false }


res2.value.then(res=>{

  console.log(res); // 2

})


let res3 = gen.next();

console.log(res3);  // { value: 'ending', done: true }


let res4 = gen.next();

console.log(res4); // { value: undefined, done: true }


Generator实现async函数


// 接受一个Generator函数作为参数

function spawn(genF) {

  // 返回一个函数

  return function() {

    // 生成指针对象

    const gen = genF.apply(this, arguments);

    // 返回一个promise

    return new Promise((resolve, reject) => {

          // key有next和throw两种取值,分别对应了gen的next和throw方法

          // arg参数则是用来把promise resolve出来的值交给下一个yield

          function step(key, arg) {

            let result;


                // 监控到错误 就把promise给reject掉 外部通过.catch可以获取到错误

                try {

                  result = gen[key](arg)

                } catch (error) {

                  return reject(error)

                }


                // gen.next() 返回 { value, done } 的结构

                const { value, done } = result;


                if (done) {

                      // 如果已经完成了 就直接resolve这个promise

                      return resolve(value)

                } else {

                      // 除了最后结束的时候外,每次调用gen.next()

                      return Promise.resolve(

                        // 这个value对应的是yield后面的promise

                        value

                      ).then((val)=>step("next", val),(err) =>step("throw", err))

                }

          }

          step

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

推荐阅读更多精彩内容