async await 运用+根底+原理

async/await用法

其实你要完成一个东西之前,最好是先搞清楚这两样东西

这个东西有什么用?
这个东西是怎样用的?

有什么用?

async/await的用处便是:用同步办法,履行异步操作,怎样说呢?举个比方
比方我现在有一个需求:先恳求完接口1,再去恳求接口2,咱们通常会这么做

function request(num) { // 模仿接口恳求
  return new Promise(resolve => {
    setTimeout(() => {
      resolve(num * 2)
    }, 1000)
  })
}
request(1).then(res1 => {
  console.log(res1) // 1秒后 输出 2
  request(2).then(res2 => {
    console.log(res2) // 2秒后 输出 4
  })
})

或许我现在又有一个需求:先恳求完接口1,再拿接口1回来的数据,去作为接口2的恳求参数,那咱们也能够这么做

request(5).then(res1 => {
  console.log(res1) // 1秒后 输出 10
  request(res1).then(res2 => {
    console.log(res2) // 2秒后 输出 20
  })
})

其实这么做是没问题的,可是假如嵌套的多了,不免有点不雅观,这个时分就能够用async/await来解决了

async function fn () {
  const res1 = await request(5)
  const res2 = await request(res1)
  console.log(res2) // 2秒后输出 20
}
fn()

是怎样用?

仍是用刚刚的比方
需求一:

async function fn () {
  await request(1)
  await request(2)
  // 2秒后履行完
}
fn()

需求二:

async function fn () {
  const res1 = await request(5)
  const res2 = await request(res1)
  console.log(res2) // 2秒后输出 20
}
fn()

其实就类似于日子中的排队,咱们日子中排队买东西,肯定是要上一个人买完,才轮到下一个人。而上面也一样,在async函数中,await规则了异步操作只能一个一个排队履行,然后到达用同步办法,履行异步操作的作用,这儿留意了:await只能在async函数中运用,否则会报错哦
刚刚上面的比方await后边都是跟着异步操作Promise,那假如不接Promise会怎样样呢?

function request(num) { // 去掉Promise
  setTimeout(() => {
    console.log(num * 2)
  }, 1000)
}
async function fn() {
  await request(1) // 2
  await request(2) // 4
  // 1秒后履行完  一起输出
}
fn()

能够看出,假如await后边接的不是Promise的话,有可能其实是达不到排队的作用的
说完await,咱们聊聊async吧,async是一个位于function之前的前缀,只需async函数中,才干运用await。那async履行完是回来一个什么东西呢?

async function fn () {}
console.log(fn) // [AsyncFunction: fn]
console.log(fn()) // Promise{<fulfilled>: undefined}

能够看出,async函数履行完会主动回来一个状态为fulfilled的Promise(不一定是成功态,可是肯定是个promise),也便是成功状态,可是值却是undefined,那要怎样才干使值不是undefined呢?很简略,函数有return回来值就行了

async function fn (num) {
  return num
}
console.log(fn) // [AsyncFunction: fn]
console.log(fn(10)) // Promise{<fulfilled>: 10}
fn(10).then(res => console.log(res)) // 10

能够看出,此刻就有值了,而且还能运用then办法进行输出

总结

总结一下async/await的知识点

  • await只能在async函数中运用,否则会报错
  • async函数回来的是一个Promise目标,有无值看有无return值
  • await后边最好是接Promise,尽管接其他值也能到达排队作用
  • async/await作用是用同步办法,履行异步操作

什么是语法糖?

前面说了,async/await是一种语法糖,诶!很多同学就会问,啥是语法糖呢?我个人了解便是,语法糖便是一个东西,这个东西你就算不必他,你用其他手段也能到达这个东西相同的作用,可是可能就没有这个东西这么方便了。

回归正题,async/await是一种语法糖,那就阐明用其他办法其实也能够完成他的作用,咱们今天便是讲一讲怎样去完成async/await,用到的是ES6里的迭代函数——generator函数

generator函数

基本用法

generator函数跟普通函数在写法上的差异便是,多了一个星号*,而且只需在generator函数中才干运用yield,什么是yield呢,他相当于generator函数履行的半途暂停点,比方下方有3个暂停点。而怎样才干暂停后持续走呢?那就得运用到next办法,next办法履行后会回来一个目标,目标中有value 和 done两个特点

  • value:暂停点后边接的值,也便是yield后边接的值
  • done:是否generator函数已走完,没走完为false,走完为true
function* gen() {
  yield 1
  yield 2
  yield 3
}
const g = gen()
console.log(g.next()) // { value: 1, done: false }
console.log(g.next()) // { value: 2, done: false }
console.log(g.next()) // { value: 3, done: false }
console.log(g.next()) // { value: undefined, done: true }

能够看到最终一个是undefined,这取决于你generator函数有无回来值

function* gen() {
  yield 1
  yield 2
  yield 3
  return 4
}
const g = gen()
console.log(g.next()) // { value: 1, done: false }
console.log(g.next()) // { value: 2, done: false }
console.log(g.next()) // { value: 3, done: false }
console.log(g.next()) // { value: 4, done: true }

yield后边接函数

yield后边接函数的话,到了对应暂停点yield,会马上履行此函数,而且该函数的履行回来值,会被作为此暂停点目标的value

function fn(num) {
  console.log(num)
  return num
}
function* gen() {
  yield fn(1)
  yield fn(2)
  return 3
}
const g = gen()
console.log(g.next()) 
// 1
// { value: 1, done: false }
console.log(g.next())
// 2
//  { value: 2, done: false }
console.log(g.next()) 
// { value: 3, done: true }

yield后边接Promise

前面说了,函数履行回来值会作为暂停点目标的value值,那么下面比方就能够了解了,前两个的value都是pending状态的Promise目标

function fn(num) {
  return new Promise(resolve => {
    setTimeout(() => {
      resolve(num)
    }, 1000)
  })
}
function* gen() {
  yield fn(1)
  yield fn(2)
  return 3
}
const g = gen()
console.log(g.next()) // { value: Promise { <pending> }, done: false }
console.log(g.next()) // { value: Promise { <pending> }, done: false }
console.log(g.next()) // { value: 3, done: true }

其实咱们想要的成果是,两个Promise的成果1 和 2,那怎样做呢?很简略,运用Promise的then办法就行了

const g = gen()
const next1 = g.next()
next1.value.then(res1 => {
  console.log(next1) // 1秒后输出 { value: Promise { 1 }, done: false }
  console.log(res1) // 1秒后输出 1
  const next2 = g.next()
  next2.value.then(res2 => {
    console.log(next2) // 2秒后输出 { value: Promise { 2 }, done: false }
    console.log(res2) // 2秒后输出 2
    console.log(g.next()) // 2秒后输出 { value: 3, done: true }
  })
})

next函数传参

generator函数能够用next办法来传参,而且能够经过yield来接纳这个参数,留意两点

  • 第一次next传参是没用的,只需从第二次开端next传参才有用
  • next传值时,要记住次序是,先右边yield,后左面接纳参数
function* gen() {
  const num1 = yield 1
  console.log(num1)
  const num2 = yield 2
  console.log(num2)
  return 3
}
const g = gen()
console.log(g.next()) // { value: 1, done: false }
console.log(g.next(11111))
// 11111
//  { value: 2, done: false }
console.log(g.next(22222)) 
// 22222
// { value: 3, done: true }

Promise+next传参

前面讲了

yield后边接Promise
next函数传参

那这两个组合起来会是什么样呢?

function fn(nums) {
  return new Promise(resolve => {
    setTimeout(() => {
      resolve(nums * 2)
    }, 1000)
  })
}
function* gen() {
  const num1 = yield fn(1)
  const num2 = yield fn(num1)
  const num3 = yield fn(num2)
  return num3
}
const g = gen()
const next1 = g.next()
next1.value.then(res1 => {
  console.log(next1) // 1秒后一起输出 { value: Promise { 2 }, done: false }
  console.log(res1) // 1秒后一起输出 2
  const next2 = g.next(res1) // 传入前次的res1
  next2.value.then(res2 => {
    console.log(next2) // 2秒后一起输出 { value: Promise { 4 }, done: false }
    console.log(res2) // 2秒后一起输出 4
    const next3 = g.next(res2) // 传入前次的res2
    next3.value.then(res3 => {
      console.log(next3) // 3秒后一起输出 { value: Promise { 8 }, done: false }
      console.log(res3) // 3秒后一起输出 8
       // 传入前次的res3
      console.log(g.next(res3)) // 3秒后一起输出 { value: 8, done: true }
    })
  })
})

完成 async await

其实上方的generator函数的Promise+next传参,就很像async/await了,差异在于

  • gen函数履行回来值不是Promise,asyncFn履行回来值是Promise
  • gen函数需求履行相应的操作,才干等同于asyncFn的排队作用
  • gen函数履行的操作是不完善的,由于并不确认有几个yield,不确认会嵌套几回

那咱们怎样办呢?咱们能够封装一个高阶函数,接纳一个generator函数,并经过一系列处理,回来一个具有async函数功用的函数

function generatorToAsync(generatorFn) {
  // 经过一系列处理
  return 具有async函数功用的函数
}

回来值Promise

之前咱们说到,async函数的履行回来值是一个Promise,那咱们要怎样完成相同的成果呢

function* gen() {
}
const asyncFn = generatorToAsync(gen)
console.log(asyncFn()) // 期望这儿输出 Promise

其实很简略,generatorToAsync函数里做一下处理就行了

function* gen() {
}
function generatorToAsync (generatorFn) {
  return function () {
    return new Promise((resolve, reject) => {
    })
  }
}
const asyncFn = generatorToAsync(gen)
console.log(asyncFn()) // Promise

参加一系列操作

咱们把之前的处理代码,参加generatorToAsync函数中

function fn(nums) {
  return new Promise(resolve => {
    setTimeout(() => {
      resolve(nums * 2)
    }, 1000)
  })
}
function* gen() {
  const num1 = yield fn(1)
  const num2 = yield fn(num1)
  const num3 = yield fn(num2)
  return num3
}
function generatorToAsync(generatorFn) {
  return function () {
    return new Promise((resolve, reject) => {
      const g = generatorFn()
      const next1 = g.next()
      next1.value.then(res1 => {
        const next2 = g.next(res1) // 传入前次的res1
        next2.value.then(res2 => {
          const next3 = g.next(res2) // 传入前次的res2
          next3.value.then(res3 => {
            // 传入前次的res3
            resolve(g.next(res3).value)
          })
        })
      })
    })
  }
}
const asyncFn = generatorToAsync(gen)
asyncFn().then(res => console.log(res)) // 3秒后输出 8

能够发现,咱们其完成已完成了以下的async/await的成果了

async function asyncFn() {
  const num1 = await fn(1)
  const num2 = await fn(num1)
  const num3 = await fn(num2)
  return num3
}
asyncFn().then(res => console.log(res)) // 3秒后输出 8

完善代码

上面的代码其实都是死代码,由于一个async函数中可能有2个await,3个await,5个await
,其实await的个数是不确认的。相同类比,generator函数中,也可能有2个yield,3个yield,5个yield,所以咱们得把代码写成活的才行

function generatorToAsync(generatorFn) {
  return function() {
    const gen = generatorFn.apply(this, arguments) // gen有可能传参
    // 回来一个Promise
    return new Promise((resolve, reject) => {
      function go(key, arg) {
        let res
        try {
          res = gen[key](arg) // 这儿有可能会履行回来reject状态的Promise
        } catch (error) {
          return reject(error) // 报错的话会走catch,直接reject
        }
        // 解构取得value和done
        const { value, done } = res
        if (done) {
          // 假如done为true,阐明走完了,进行resolve(value)
          return resolve(value)
        } else {
          // 假如done为false,阐明没走完,还得持续走
          // value有可能是:常量,Promise,Promise有可能是成功或许失败
          return Promise.resolve(value).then(val => go('next', val), err => go('throw', err))
        }
      }
      go("next") // 第一次履行
    })
  }
}
const asyncFn = generatorToAsync(gen)
asyncFn().then(res => console.log(res))

这样的话,无论是多少个yield都会排队履行了,咱们把代码写成活的了

示例

async/await版别

async function asyncFn() {
  const num1 = await fn(1)
  console.log(num1) // 2
  const num2 = await fn(num1)
  console.log(num2) // 4
  const num3 = await fn(num2)
  console.log(num3) // 8
  return num3
}
const asyncRes = asyncFn()
console.log(asyncRes) // Promise
asyncRes.then(res => console.log(res)) // 8

运用generatorToAsync函数的版别

function* gen() {
  const num1 = yield fn(1)
  console.log(num1) // 2
  const num2 = yield fn(num1)
  console.log(num2) // 4
  const num3 = yield fn(num2)
  console.log(num3) // 8
  return num3
}
const genToAsync = generatorToAsync(gen)
const asyncRes = genToAsync()
console.log(asyncRes) // Promise
asyncRes.then(res => console.log(res)) // 8

参考
/post/700703…

async await ssh

常常有人说async函数是generator函数的语法糖,那么到底是怎样样一个糖呢?让咱们来一层层的剥开它的糖衣。
有的同学想说,已然用了generator函数何必还要完成async呢?
这篇文章的意图便是带咱们了解清楚async和generator之间到底是怎么相互协作,管理异步的。

// 示例

const getData = () => new Promise(resolve => setTimeout(() => resolve("data"), 1000))
async function test() {
const data = await getData()
console.log('data: ', data);
const data2 = await getData()
console.log('data2: ', data2);
return 'success'
}
// 这样的一个函数 应该再1秒后打印data 再过一秒打印data2 最终打印success
test().then(res => console.log(res))

关于这个简略的事例来说,假如咱们把它用generator函数表达,会是怎样样的呢?

function* testG() {
// await被编译成了yield
const data = yield getData()
console.log('data: ', data);
const data2 = yield getData()
console.log('data2: ', data2);
return 'success'
}
// 咱们知道,generator函数是不会主动履行的,每一次调用它的next办法,会停留在下一个yield的位置。
// 利用这个特性,咱们只需编写一个主动履行的函数,就能够让这个generator函数完全完成async函数的功用。
const getData = () => new Promise(resolve => setTimeout(() => resolve("data"), 1000))
var test = asyncToGenerator(
function* testG() {
// await被编译成了yield
const data = yield getData()
console.log('data: ', data);
const data2 = yield getData()
console.log('data2: ', data2);
return 'success'
}
)
test().then(res => console.log(res))

那么大体上的思路现已确认了,
asyncToGenerator承受一个generator函数,回来一个promise,
要害就在于,里面用yield来区分的异步流程,应该怎么主动履行

假如是手动履行

在编写这个函数之前,咱们先模仿手动去调用这个generator函数去一步步的把流程走完,有助于后边的思考。

function* testG() {
// await被编译成了yield
const data = yield getData()
console.log('data: ', data);
const data2 = yield getData()
console.log('data2: ', data2);
return 'success'
}
// 咱们先调用testG生成一个迭代器
// 回来了一个迭代器
var gen = testG()
// 然后开端履行第一次next
// 第一次调用next 停留在第一个yield的位置
// 回来的promise里 包含了data需求的数据
var dataPromise = gen.next()
// 这儿回来了一个promise,便是第一次getData()所回来的promise,留意
const data = yield getData()
// 这段代码要切割成左右两部分来看,第一次调用next,其实仅仅停留在了yield getData()这儿,
// data的值并没有被确认。
// 那么什么时分data的值会被确认呢?
// 下一次调用next的时分,传的参数会被作为上一个yield前面承受的值
// 也便是说,咱们再次调用gen.next('这个参数才会被赋给data变量')的时分
// data的值才会被确认为'这个参数才会被赋给data变量'
gen.next('这个参数才会被赋给data变量')
// 然后这儿的data才有值
const data = yield getData()
// 然后打印出data
console.log('data: ', data);
// 然后持续走到下一个yield
const data2 = yield getData()
// 然后往下履行,直到遇到下一个yield,持续这样的流程...
// 这是generator函数规划的一个比较难了解的点,可是为了完成咱们的目标,仍是得去学习它~
// 凭借这个特性,假如咱们这样去控制yield的流程,是不是就能完成异步串行了?
function* testG() {
// await被编译成了yield
const data = yield getData()
console.log('data: ', data);
const data2 = yield getData()
console.log('data2: ', data2);
return 'success'
}
var gen = testG()
var dataPromise = gen.next()
console.log(dataPromise)
dataPromise.value.then((value1) => {
// data1的value被拿到了 持续调用next而且传递给data
var data2Promise = gen.next(value1)
// console.log('data: ', data);
// 此刻就会打印出data
data2Promise.value.then((value2) => {
// data2的value拿到了 持续调用next而且传递value2
gen.next(value2)
// console.log('data2: ', data2);
// 此刻就会打印出data2
})
})
// 这样的一个看着像callback hell的调用,就能够让咱们的generator函数把异步组织的明明白白。

完成

// 有了这样的思路,完成这个高阶函数就变得很简略了。
// 先全体看一下结构,有个印象,然后咱们逐行注释讲解。

function asyncToGenerator (generatorFunc) {
    return function () {
        const gen = generatorFunc.apply(this, arguments)
        return new Promise((resolve, reject) => {
            function step(key, arg) {
                let generatorResult
                try {  
                    generatorResult = gen[key](arg)
                } catch {
                    return reject(error)
                }
                const {value, done} = generatorResult
                if (done) {
                    return resolve(value)
                } else {
                    return Promise.resolve(value).then(val => step('next', val), err => step('throw',err))
                }
            }
            step('next')
        })
    }
}

接下来逐行讲解。

function asyncToGenerator(generatorFunc) {
// 回来的是一个新的函数
return function() {
// 先调用generator函数 生成迭代器
// 对应 var gen = testG()
const gen = generatorFunc.apply(this, arguments)
// 回来一个promise 由于外部是用.then的办法 或许await的办法去运用这个函数的回来值的
// var test = asyncToGenerator(testG)
// test().then(res => console.log(res))
return new Promise((resolve, reject) => {
// 内部界说一个step函数 用来一步一步的跨过yield的阻碍
// key有next和throw两种取值,分别对应了gen的next和throw办法
// arg参数则是用来把promise resolve出来的值交给下一个yield
function step(key, arg) {
let generatorResult
// 这个办法需求包裹在try catch中
// 假如报错了 就把promise给reject掉 外部经过.catch能够获取到错误
try {
generatorResult = gen[key](arg)
} catch (error) {
return reject(error)
}
// gen.next() 得到的成果是一个 { value, done } 的结构
const { value, done } = generatorResult
if (done) {
// 假如现已完成了 就直接resolve这个promise
// 这个done是在最终一次调用next后才会为true
// 以本文的比方来说 此刻的成果是 { done: true, value: 'success' }
// 这个value也便是generator函数最终的回来值
return resolve(value)
} else {
// 除了最终完毕的时分外,每次调用gen.next()
// 其实是回来 { value: Promise, done: false } 的结构,
// 这儿要留意的是Promise.resolve能够承受一个promise为参数
// 而且这个promise参数被resolve的时分,这个then才会被调用
return Promise.resolve(
// 这个value对应的是yield后边的promise
value
).then(
// value这个promise被resove的时分,就会履行next
// 而且只需done不是true的时分 就会递归的往下解开promise
// 对应gen.next().value.then(value => {
// gen.next(value).value.then(value2 => {
// gen.next()
//
// // 此刻done为true了 整个promise被resolve了
// // 最外部的test().then(res => console.log(res))的then就开端履行了
// })
// })
function onResolve(val) {
step("next", val)
},
// 假如promise被reject了 就再次进入step函数
// 不同的是,这次的try catch中调用的是gen.throw(err)
// 那么自然就被catch到 然后把promise给reject掉啦
function onReject(err) {
step("throw", err)
},
)
}
}
step("next")
})
}
}

// 本文用最简略的办法完成了asyncToGenerator这个函数,这是babel编译async函数的中心,当然在babel中,generator函数也被编译成了一个很原始的方式,本文咱们直接以generator替代。

ES6 系列之 Babel 将 Async 编译成了什么姿态

本文便是简略介绍下 Async 语法编译后的代码。
Async

const fetchData = (data) => new Promise((resolve) => setTimeout(resolve, 1000, data + 1))
const fetchValue = async function () {
var value1 = await fetchData(1);
var value2 = await fetchData(value1);
var value3 = await fetchData(value2);
console.log(value3)
};
fetchValue();
// 大约 3s 后输出 4

Babel

// 咱们直接在 Babel 官网的 Try it out 张贴上述代码,然后检查代码编译成什么姿态:

"use strict";
function _asyncToGenerator(fn) {
return function() {
var gen = fn.apply(this, arguments);
return new Promise(function(resolve, reject) {
function step(key, arg) {
try {
var info = gen[key](arg);
var value = info.value;
} catch (error) {
reject(error);
return;
}
if (info.done) {
resolve(value);
} else {
return Promise.resolve(value).then(
function(value) {
step("next", value);
},
function(err) {
step("throw", err);
}
);
}
}
return step("next");
});
};
}
var fetchData = function fetchData(data) {
return new Promise(function(resolve) {
return setTimeout(resolve, 1000, data + 1);
});
};
var fetchValue = (function() {
var _ref = _asyncToGenerator(
/*#__PURE__*/ regeneratorRuntime.mark(function _callee() {
var value1, value2, value3;
return regeneratorRuntime.wrap(
function _callee$(_context) {
while (1) {
switch ((_context.prev = _context.next)) {
case 0:
_context.next = 2;
return fetchData(1);
case 2:
value1 = _context.sent;
_context.next = 5;
return fetchData(value1);
case 5:
value2 = _context.sent;
_context.next = 8;
return fetchData(value2);
case 8:
value3 = _context.sent;
console.log(value3);
case 10:
case "end":
return _context.stop();
}
}
},
_callee,
this
);
})
);
return function fetchValue() {
return _ref.apply(this, arguments);
};
})();
fetchValue();

_asyncToGenerator

这次咱们重点来看看 _asyncToGenerator 函数:

function _asyncToGenerator(fn) {
return function() {
var gen = fn.apply(this, arguments);
return new Promise(function(resolve, reject) {
function step(key, arg) {
try {
var info = gen[key](arg);
var value = info.value;
} catch (error) {
reject(error);
return;
}
if (info.done) {
resolve(value);
} else {
return Promise.resolve(value).then(
function(value) {
step("next", value);
},
function(err) {
step("throw", err);
}
);
}
}
return step("next");
});
};
}

以上这段代码主要是用来完成 generator 的主动履行以及回来 Promise。
当咱们履行 fetchValue() 的时分,履行的其实便是 _asyncToGenerator 回来的这个匿名函数,在匿名函数中,咱们履行了

var gen = fn.apply(this, arguments);
// 这一步就相当于履行 Generator 函数,举个比方:
function* helloWorldGenerator() {
yield 'hello';
yield 'world';
return 'ending';
}
var hw = helloWorldGenerator();

var gen = fn.apply(this, arguments) 就相当于 var hw = helloWorldGenerator();,回来的 gen 是一个具有 next()、throw()、return() 办法的目标。
// 然后咱们回来了一个 Promise 目标,在 Promise 中,咱们履行了 step(“next”),step 函数中会履行:

try {
var info = gen[key](arg);
var value = info.value;
} catch (error) {
reject(error);
return;
}
// step("next") 就相当于 var info = gen.next(),回来的 info 目标是一个具有 value 和 done 特点的目标:
// {value: Promise, done: false}
// 接下来又会履行:
if (info.done) {
resolve(value);
} else {
return Promise.resolve(value).then(
function(value) {
step("next", value);
},
function(err) {
step("throw", err);
}
);
}

value 此刻是一个 Promise,Promise.resolve(value) 仍然会回来这个 Promise,咱们给这个 Promise 添加了一个 then 函数,用于在 Promise 有成果时履行,有成果时又会履行 step(“next”, value),然后使得 Generator 持续履行,直到 info.done 为 true,才会 resolve(value)。

// 不完整但可用的代码

(function() {
var ContinueSentinel = {};
var mark = function(genFun) {
var generator = Object.create({
next: function(arg) {
return this._invoke("next", arg);
}
});
genFun.prototype = generator;
return genFun;
};
function wrap(innerFn, outerFn, self) {
var generator = Object.create(outerFn.prototype);
var context = {
done: false,
method: "next",
next: 0,
prev: 0,
sent: undefined,
abrupt: function(type, arg) {
var record = {};
record.type = type;
record.arg = arg;
return this.complete(record);
},
complete: function(record, afterLoc) {
if (record.type === "return") {
this.rval = this.arg = record.arg;
this.method = "return";
this.next = "end";
}
return ContinueSentinel;
},
stop: function() {
this.done = true;
return this.rval;
}
};
generator._invoke = makeInvokeMethod(innerFn, context);
return generator;
}
function makeInvokeMethod(innerFn, context) {
var state = "start";
return function invoke(method, arg) {
if (state === "completed") {
return { value: undefined, done: true };
}
context.method = method;
context.arg = arg;
while (true) {
state = "executing";
if (context.method === "next") {
context.sent = context._sent = context.arg;
}
var record = {
type: "normal",
arg: innerFn.call(self, context)
};
if (record.type === "normal") {
state = context.done ? "completed" : "yield";
if (record.arg === ContinueSentinel) {
continue;
}
return {
value: record.arg,
done: context.done
};
}
}
};
}
window.regeneratorRuntime = {};
regeneratorRuntime.wrap = wrap;
regeneratorRuntime.mark = mark;
})();
"use strict";
function _asyncToGenerator(fn) {
return function() {
var gen = fn.apply(this, arguments);
return new Promise(function(resolve, reject) {
function step(key, arg) {
try {
var info = gen[key](arg);
var value = info.value;
} catch (error) {
reject(error);
return;
}
if (info.done) {
resolve(value);
} else {
return Promise.resolve(value).then(
function(value) {
step("next", value);
},
function(err) {
step("throw", err);
}
);
}
}
return step("next");
});
};
}
var fetchData = function fetchData(data) {
return new Promise(function(resolve) {
return setTimeout(resolve, 1000, data + 1);
});
};
var fetchValue = (function() {
var _ref = _asyncToGenerator(
/*#__PURE__*/
regeneratorRuntime.mark(function _callee() {
var value1, value2, value3;
return regeneratorRuntime.wrap(
function _callee$(_context) {
while (1) {
switch ((_context.prev = _context.next)) {
case 0:
_context.next = 2;
return fetchData(1);
case 2:
value1 = _context.sent;
_context.next = 5;
return fetchData(value1);
case 5:
value2 = _context.sent;
_context.next = 8;
return fetchData(value2);
case 8:
value3 = _context.sent;
console.log(value3);
case 10:
case "end":
return _context.stop();
}
}
},
_callee,
this
);
})
);
return function fetchValue() {
return _ref.apply(this, arguments);
};
})();
fetchValue();

上边两个章节中最重要的便是这个

function _asyncToGenerator(fn) {
  return function() {
    var gen = fn.apply(this, arguments);
    return new Promise(function(resolve, reject) {
      function step(key, arg) {
        try {
        var info = gen[key](arg);
        var value = info.value;
        } catch (error) {
        reject(error);
        return;
        }
        if (info.done) {
        resolve(value);
        } else {
        return Promise.resolve(value).then(
        function(value) {
        step("next", value);
        },
        function(err) {
        step("throw", err);
        }
        );
      }
    }
    return step("next");
  });
  };
}