发展历程

Js 异步的发展历程,从简陋,心智模型负担较重的语法范式到非常近似同步的语法,一步步走来,对前端开发者来说,影响巨大。他的发展历程简单来说如下所示:

callback -> deferred(promise) -> promise -> generator + promise -> async/await

什么是 callback

如果一个函数无法立即返回 value,而是经过一段不可预测的行为时间之后(副作用),才能得到 value 我们要如何做才能获得 value?

function ordinary () {
  const i = value
  // ...
  return value
}
function sideEffect () {
  const value = 1
  setTimeout(() => {
    return value
  })
}
console.log(ordinary()) // 1console.log(sideEffect()) // undefined复制代码
function sideEffect (callback) {
  const value = 1
  setTimeout(() => {
    // ...
    callback(value)
  })
}

sideEffect(value => {
  console.log(value) // 1
})复制代码

从上面的代码可以看到,callback 让我们拥有了获取不可预测行为结果的能力,这得益于 JavaScript 函数是一等公民。

社区的方案

但是 callback 带来了很严重的语法层面上的问题。

callback 回调地狱

getData(function (a) {
  getMoreData(a, function (b) {
    getMoreData(b, function (c) {
      getMoreData(c, function (d) {
        // ...
      })
    })
  })
})复制代码
// 需要合理的封装和简化,这需要开发人员自身的水平和认知决定function fn (a, cb) {
  getMoreData(a, function (b) {
    getMoreData(b, function (c) {
      getMoreData(c, function (d) {
        cb(d)
      })
    })
  })
}
// 通过层层封装,抽象出模块和通用的类来保证代码是浅层的
getData(function (a) {
  fn(a, function (d) {
    // ...
  })
})复制代码

bug 在封装和简化的过程中很容易产生。

为什么是 promise 胜出

社区陆续出来了 promise 和类 promise 的方案。JQuery1.5 中就有了 deferred 的概念。

通过 promise 的形式重写。

// 将 callback 变成了一种扁平化的结构
// 相对于 callback 是更加同步的思维将代码结构铺开来

getData()
  .then(getMoreData)
  .then(getMoreData)
  .then(getMoreData)
  .then(function (d) {
    // ...
  })复制代码

社区有几种不同的方案,为什么最后 es6 选择了 promise 方案。

// deferred
deferred.promise.then(v => console.log(v))
setTimeout(() => {
  deferred.resolve('tao')
}, 500)

// promise
const p = new Promise(resolve => {
  setTimeout(() => {
    resolve('tao')
  }, 500)
})
p.then(v => console.log(v))复制代码

看看下面这俩例子,看看为啥最终选择的是 promise。

// 不会捕捉到错误
deferred.promise.catch(reason => console.log(reason))

setTimeout(() => {
  throw 'error'
})复制代码
// 必须用 try catch 然后通过 deferrd.reject 触发
deferred.promise.catch(reason => console.log(reason))

setTimeout(() => {
  try {
    throw 'error'
  } catch (err) {
    deferred.reject(err)
  }
})复制代码
// promise 由于是自执行,自动捕捉异常
const p = new Promise(() => {
  throw 'error'
})

p.catch(reason => console.log(reason))复制代码

总结

  1. promise 首先应该是一个异步流程控制的解决方案,流程控制包括了正常的数据流和异常流程处理

  2. deferred 的方式存在一个致命的缺陷

  3. 就是 promise 链的第一个 promise(deferred.promise)的触发阶段抛出的异常是不交由 promise 自动处理的

deferred 对象

deferred 对象其实就是一个发布/订阅模式

function createDeferred () {
  let resolve, reject

  const promise = new Promise((_resolve, _reject) => {
    resolve = _resolve
    reject = _reject
  })
  return { promise, resolve, reject }
}复制代码
Promise + generator

promise 链式调用的语法还是不够同步,怎么办?看看下面这俩例子,感受语法的变化。

const getData = () => {
  return new Promise(resolve => resolve(1))
}
const getMoreData = value => {
  return value + 1
}

getData()
  .then(getMoreData)
  .then(getMoreData)
  .then(getMoreData)
  .then(value => {
    console.log(value) // 4
  })复制代码

通过 generator 函数,但是需要我们手动调用 next()。

const gen = (function * () {
    const a = yield 1
    const b = yield a + 1
    const c = yield b + 1
    const d = yield c + 1return d
})()
const a = gen.next()
const b = gen.next(a.value)
const c = gen.next(b.value)
const d = gen.next(c.value)
console.log(d.value) // 4复制代码
自执行函数

自己手动封装一个自动执行 next 的函数。

function co (fn, ...args) {
  return new Promise((resolve, reject) => {
    const gen = fn(...args)

    function next (result) { ... }

    function onFulfilled (res) { ... }

    function onRejected (err) { ... }

    onFulfilled()
  })
}复制代码
// 自动调用 gen.next()
// 然后调用 next() 将结果传入到 generator 对象内部
function onFulfilled (res) {
  let result

  try {
    result = gen.next(res)
    next(result)
  } catch (err) {
    return reject(err)
  }
}复制代码
// 发生错误调用 gen.throw()
// 这可以让 generator 函数内部的 try/catch 捕获到
function onRejected (res) {
  let result

  try {
    result = gen.throw(err)
    next(result)
  } catch (err) {
    return reject(err)
  }
}复制代码
// 接受到结果后再次调用 onFulfilled
// 继续执行 generator 内部的代码
function next (result) {
  let value = result.value
  if (result.done) return resolve(value)

  // 如果是 generator 函数,等待整个 generator 函数执行完毕
  if (
      value && value.constructor && 
      value.constructor.name === 'GeneratorFunction'
  ) {
    value = co(value)
  }

  // 转为 promise
  Promise.resolve(value).then(onFulfilled, onRejected)
}复制代码

看看效果

const ret = co(function * () {
  const a = yield 1
  const b = yield a + 1
  const c = yield b + 1
  const d = yield c + 1
  return d
})
ret.then(v => console.log(v)) // 4复制代码

结合 promise。

const fn = v => {
  return new Promise(resolve => {
    setTimeout(() => resolve(v), 200)
  })
}
const ret = co(function * () {
  const a = yield fn(1)
  console.log(a) // 1

  const b = yield fn(a + 1)
  console.log(b) // 2

  const c = yield fn(b + 1)
  console.log(c) // 3

  const d = yield fn(c + 1)
  console.log(d) // 4

  return d
})
ret.then(v => console.log(v)) // 4复制代码

error 的处理

// 错误都能被捕捉
const ret = co(function * () {
  try {
    throw 'errorOne'
  } catch (err) {
    console.log(err) // errorOne
    throw 'errorTwo'
  }
})

ret.catch(err => console.log(err)) // errorTwo复制代码

看起来是不是比 promise 的写法“同步”多了。够了吗?当然还不够。

总结

  1. 在这个名叫 co 的自执行函数里面

  2. onFulfilled 调用 next

  3. next 调用 onFulfilled

  4. 这样就形成一个自执行器,只有当代码全部执行完毕后才会终止

async/await 语法糖

ES2017 标准引入了 async 函数,使得异步操作变得更加方便。async 函数是什么?它就是 Generator 函数的语法糖。

做个对比,理解一下为什么说是语法糖

const ret = (async function () {
  const a = await fn(1)
  const b = await fn(a + 1)
  const c = await fn(b + 1)
  const d = await fn(c + 1)
  return d
})()
ret.then(v => console.log(v))复制代码
const ret = co(function * () {
  const a = yield fn(1)
  const b = yield fn(a + 1)
  const c = yield fn(b + 1)
  const d = yield fn(c + 1)
  return d
})
ret.then(v => console.log(v))复制代码

Async 函数

async 函数作为被纳入 ES 规范的语法,自然会随着引擎不断优化迭代,肯定会比我们自己写执行器要更好,我们稍微探究一下 async 函数。

// 将会打印 'into'
// 这表明 async 函数会在 promise 后面添加 p.then() 的行为
// 这无关 promise 是哪一种实现(theable 也是可以的)
const p = {
  then (resolve, reject) {
    console.log('into')
    setTimeout(() => resolve('tao'), 1000)
    // reject('err')
  }
}

(async function () {
  try {
    const v = await p
    console.log(v)
  } catch (err) {
    console.log(err)
  }
})()复制代码

我们的写法,最终演变成了这样。

xhr.get('xx', data, res => {
  console.log(res)
})复制代码
const res = await xhr.get('xx', data)
console.log(res)复制代码
异步演变总结
  1. 我们理解了为什么需要 callback

  2. 也知道了 callback 带来的问题是什么

  3. 社区给了解决方案并最终被规范所接纳

  4. promise + generator 带来更极致的异步编程体验

  5. async/await 语法糖更加强化这一体验

  6. 这一系列的变化让我们对异步的控制流程加强了很多,更加同步直观的语法,带来更少的维护负担和更少的 bug

co/next

我们只解读 next 函数。

function next(ret) {
  if (ret.done) return resolve(ret.value)
  var value = toPromise.call(ctx, ret.value)
  if (value && isPromise(value)) return value.then(onFulfilled, onRejected)

  return onRejected(new TypeError(
    'You may only yield a function, promise, generator, array, or object, '
    + 'but the following object was passed: "' + String(ret.value) + '"')
  )
}复制代码
function toPromise(obj) {
  if (!obj) return obj
  if (isPromise(obj)) return obj
  if (isGeneratorFunction(obj) || isGenerator(obj)) return co.call(this, obj)
  if ('function' == typeof obj) return thunkToPromise.call(this, obj)
  if (Array.isArray(obj)) return arrayToPromise.call(this, obj)
  if (isObject(obj)) return objectToPromise.call(this, obj)

  return obj
}复制代码

通过 next 和 toPromsie 函数源码可以知道,co 只支持以下几种数据类型:

  1. array

  2. object

  3. promise

  4. generator

  5. function(thunk function 我们不做讨论)

array

// 数组种的所有 item 都做了 promise 的过滤
function arrayToPromise (obj) {
  return Promise.all(obj.map(toPromise, this))
}复制代码

object

function objectToPromise(obj) {
  var results = new obj.constructor()
  var keys = Object.keys(obj)
  var promises = []

  for (var i = 0; i < keys.length; i++) {
    var key = keys[i]
    var promise = toPromise.call(this, obj[key])

    // 这里将 object 中是 promise 的 item 筛选出来,通过 promise.all 来处理
    if (promise && isPromise(promise)) defer(promise, key)
    else results[key] = obj[key]
  }

  return Promise.all(promises).then(function () {
    return results
  })

  function defer(promise, key) {
    // js 引擎喜欢稳定的对象结构,所有预先定义(也告诉我们少用 delete 语句)
    results[key] = undefined
    promises.push(promise.then(function (res) {
      results[key] = res
    }))
  }
}复制代码

看看 demo

objectToPromise({
  a: 1,
  b: [2, 3],
  c: new Promise(resolve => {
    setTimeout(() => resolve(1), 500)
  })
}).then(res => {
  console.log(res) // { a: 1, b: [2, 3], c: 1 }
})复制代码

其实有个疑问是,为什么要使用下面的这种方法判断是不是 promise。

function isPromise (obj) {
  return 'function' == typeof obj.then
}复制代码
Theable 与鸭子模型

如果一个对象 x 有一个 then 方法,那么 x 就是一个 thenable,then 会被立即调用,传入参数 resolve和 reject,并绑定 x 作为 this,而 thenable 就是 promise 的 鸭子类型

所以,才有了下面的这种写法

Promise.resolve({
  then (resolve, reject) {
    resolve(1)
  }
})
.then(res => {
  console.log(res) // 1
})复制代码

但是为啥要使用鸭子类呢?

我们不需要判断是不是一个 promsie,只需要判断像不像一个 promise。无论是你自己写 promsie,第三方库的 promise,还是 js 引擎实现的 promsie。这带来了良好的兼容性。

co.wrap
// 创建一个高阶函数
co.wrap = function (fn) {
  createPromise.__generatorFunction__ = fn // 单元测试用
  return createPromise
  function createPromise() {
    return co.call(this, fn.apply(this, arguments))
  }
}复制代码
const fn = co.wrap(function * () {})
const gen = fn()复制代码
Bug demo

如果加深了对异步副作用的认识,请注意这样的 bug。

<template>
  <div @click='getMessage'></div>
</template>

<script>
  export default {
    data: () => ({
      message: '',
    }),

    methods: {
      // 问题在于异步请求是副作用,我们无法预测这个结果在什么时间到来
      // 导致我们无法保证程序的顺序。同样也很难复现,同样的输入可能导致不同的输出
      async getMessage () {
        this.message = await fetch('xx')
      }
    }
  }
</script>复制代码

可以改成下面这样

<template>
  <div @click='getMessage'></div>
</template>
<script>
  export default {
    data: () => ({
      message: '',
      requestId: 0,
    }),

    methods: {
      // 利用闭包拒绝掉已经丢弃的副作用行为
      async getMessage () {
        const id = ++this.requestId
        const res = await fetch('xx')

        if (id !== this.requestId) return
        this.message = res
      }
    }
  }
</script>复制代码
总结
  1. co 在整个 Js 的异步发展历史中处于一个很关键的节点

  2. co 将 promise 和 generator 函数结合在一起,给了 Js 更加强大的生命力

  3. 到最后的发展阶段,我们有了控制异步行为更好的手段,这让我们能更好的结合函数式编程

  4. Js 的异步与各个平台的 event loop 息息相关,不同平台的行为可能不一致(后话)

  5. Js 的异步发展是 ES 规范中很重要的一部分,但是 es6 的发展也包含了其他,例如:

    1. 更好的数据结构(map,set,weakmap,weakset)

    2. 更好的遍历手段(for/of + iterator)

    3. 更好的数据保护机制和元编程手段(setter/getter -> proxy/reflect)

    4. 更好的 TypeArray 支持(音视频等)

    5. 当然 es 也在继续发展,未来还会有更多的新东西