使用function* / yield 实现 async/awa

2020-06-14  本文已影响0人  火锅伯南克

1.明确概念
async函数就是generator函数的语法糖。
async函数,就是将generator函数的*换成async,将yield替换成await。
2.async函数对generator的改进
(1) async内置执行器,不需要使用next()手动执行。
(2) await关键字后面的表达式返回的值可以是任何类型,只不过在解释器处理时,不是Promise类型的要转化成Promise.resolve(value)
yield关键字后面的表达式返回的值可以是任何类型。
(3) async函数返回值是Promise。返回非Promise时,async函数会把它包装成Promise返回。(Promise.resolve(value))
3.作用
异步编程的终极解决方案。

如果想彻底搞懂以上概念,需要使用function* /yield手动去实现async/await,但是实现之前,需要知道async/await和function* /yield怎么使用。
一、使用async/await

这个函数贯穿始终

//模拟异步请求
const getTime = function (time, bool){
  return new Promise(function (resolve, reject){
    setTimeout(()=> {
      bool ? resolve(time) : reject(`${time}报错了`)
    }, time)
  });
};

1.async函数声明语句

async function getData(){
  return await getTime(2000, true)
}
getData()
.then(res => {
  console.log(res) //2000
})

2.async函数表达式语句

const getData = async function (){
  return await getTime(2000, true)
}
getData()
.then(res => {
  console.log(res) //2000
})

3.IIFE使用async

( async () => {
  const data = await getTime(2000, true)
  console.log(data) //2000
})();

4.在对象上使用async

const obj1 = {
  async getData(){
    return await getTime(2000, true)
  }
};
class obj2 {
  async getData(){
    return await getTime(1000, true)
  }
}
( async () => {
  const data1 = await obj1.getData();
  const data2 = await new obj2().getData();
  console.log(data1) //2000
  console.log(data2) //1000
})();

5.async函数内,手动捕获错误&处理

async function getData(){
  return await getTime(2000, false)
}
( async () => {
  try{
    const data= await getData();
    console.log(data)
  } catch (err) {
    console.log(err) //2000报错了
  }
})();

6.async内使用同步

async function getData(time){
  await getTime(2000, true) //延迟2秒
  return await getTime(time, true)
}

( async () => {
  console.time('title')
  const data1 = await getData(1000);
  const data2 = await getData(2000);
  console.log(data1) //1000
  console.log(data2) //2000
  console.timeEnd('title')
  // title: 7015.192ms
})();

结论:两个await同步执行,整个函数运行完成7秒多

7.async内使用异步

async function getData(time){
  await getTime(2000, true) //延迟2秒
  return await getTime(time, true)
}
( async () => {
  console.time('title')
  const data1 = getData(1000);
  const data2 = getData(2000);
  //请求全部发出后,await需要等待pramise状态,
  //如果pramise是pending状态需要继续等待,
  //如果不是pending,
  //为resolve时,其参数作为 await 表达式的值
  //为rejected时,await 表达式会将其参数当做异常抛出。
  const _data1 = await data1;
  const _data2 = await data2;
  console.log(_data1) //1000
  console.log(_data2) //2000
  console.timeEnd('title')
  //title: 4015.644ms
})();

结论:两个await异步执行,整个函数运行完成4秒多。

使用Promise.all实现异步

async function getData(time){
  await getTime(2000, true) //延迟2秒
  return await getTime(time, true)
}
( async () => {
  console.time('title')
  const [data1, data2] = await Promise.all([
    getData(1000),
    getData(2000)
  ])
  console.log(data1) //1000
  console.log(data2) //2000
  console.timeEnd('title')
  //title: 4018.707ms
})();

在for循环中实现异步

async function getData(time){
  await getTime(2000, true) //延迟2秒
  return await getTime(time, true)
}
( async () => {
  console.time('title')
  let promise = [1000, 2000]
  promise = promise.map(item =>getData(item))
  for(const i of promise){
    let data = await i
    console.log(data)
    //1000
    //2000
  }
  console.timeEnd('title')
  //title: 4005.523ms
})();
二、使用function* / yield

关于function* / yield的基础方法介绍的文章
文章部分介绍了function* / yield
基础方法需要了解一下,不赘述了

三、使用function* / yield 实现 async/await 函数

实现目标:
①内部支持异步代码的同步化
②函数返回值必须是一个Promise
③yield后的值必须是Promise,不是的会被转化成Promise
④函数体内如果使用try/catch语句,能够捕获错误,且不影响函数内以下代码的执行。
⑤函数体内没有使用try/catch包裹的语句如果报错,需要在函数返回的Promise对象的catch捕获,且函数内代码执行结束。
⑥函数的参数要正常使用,且执行内部的this需要指向定义时的对象。
⑦如果执行函数内部return 后是Promise,直接取这个Promise的状态和值为己用,如果结果值依然是Promise,则继续反复,直到拿到一个非Promise值。

_async.js

/**
* 传入需要使用async的函数
* @param {function} fn
* @return {function}
*/
function _async (fn) {
  return function (...arg) {
    //实现目标6
    let gen = fn.apply(this, arg)
    //实现目标2
    return new Promise(function(resolve, reject){
      _iterator(gen, resolve, reject)
    })
  }
}
//验证是否是Promise
function isPromise(value){
  let s = Object.prototype.toString
  return s.call(value) === '[object Promise]'
}
/**
* @param {generator} gen 迭代对象
* @param {resolve} resolve Promise的成功回调
* @param {reject} reject Promise的失败回调
* @param {any} val 迭代时next需要的参数
* @param {any} err 迭代时产生的错误对象
*/
function _iterator(gen, resolve, reject, val, err){
  try{
    //gen.throw(err) 实现目标4
    var {value, done} = err ? gen.throw(err) : gen.next(val)
  }catch(err){
    //实现目标5
    return reject(err)
  }
  //实现目标7
  //这个功能Promise内部会帮我处理好
  if(done) { return resolve(value) }
  //实现目标3
  if(!isPromise(value)){
    value = Promise.resolve(value)
  }
  ////实现目标1
  value
  .then( res => {
    _iterator(gen, resolve, reject, res)
  }, err => {
    _iterator(gen, resolve, reject, null, err)
  })
}

module.exports = _async

使用

const _async = require('./_async')
//模拟异步请求
const getTime = function (time, bool){
  return new Promise(function (resolve, reject){
    setTimeout(()=> {
      bool ? resolve(time) : reject(`${time}报错了`)
    }, time)
  });
};


//模拟vue的methods
let methods =  {
  name: 'Vue',
  getPromise: _async(function* (a, b, c){
    console.log(this.name) //查看当前的活动对象
    console.log(a,b,c) //查看当前参数
    var f1 = yield getTime(1000, true);
    console.log(f1);
    var f2 = yield getTime(2000, true);
    console.log(f2);
    var f3 = yield 200;
    console.log(f3);
        //查看内部的错误捕获
    try{
      var f4 = yield getTime(1000, false);
      console.log(f4);
    }catch(err){
      console.log(err)
    }
    //打开这条注释,查看没有try的报错处理
    //throw new Error('函数内报错,')
    console.timeEnd('move')
    return getTime(2000, false)
  })
}
console.time('move')

methods.getPromise(9,8,7)
.then(res => {
  console.log(res)
}).catch(err =>{
  console.log(err)
})

大佬略,新手可以参考一下,反正我是写完之后再看function* / yield 和 async/await ,觉得很通透。

上一篇 下一篇

猜你喜欢

热点阅读