03-JavaScript-Generator异步编程

2020-10-23  本文已影响0人  低头看云

Generator

概念

形式上

Generator 函数是一个普通函数,但是有两个特征。

function* helloWorldGenerator() {
  yield 'hello';
  yield 'world';
  return 'ending';
}

var hw = helloWorldGenerator();
// --- 输出 ------
console.log(hw.next()) // { value: 'hello', done: false }
console.log(hw.next()) // { value: 'world', done: false }
console.log(hw.next()) // { value: 'ending', done: true }
console.log(hw.next()) // { value: undefined, done: true }

例子

// 执行顺序
function* test(x) {
  console.log(111)
  let a = yield x + 222

  console.log('a', a)
  let b = yield 333 + a

  console.log('b', b)
  let c = yield 44444 + b

  console.log('c', c)
}

 
let t = test(1)

console.log(t.next())
console.log(t.next())
console.log(t.next())
console.log(t.next())

/**
111
{ value: 223, done: false }
a undefined
{ value: NaN, done: false }
b undefined
{ value: NaN, done: false }
c undefined
{ value: undefined, done: true }
**/

发现除了第一个next()能正常输出{ value: 223, done: false },第二个next开始输出 a=undefined.

查阅文档后发现

yield表达式本事是没有返回值, 或者说总是返回undefined.

但是, next 方法可以带一个参数, 该参数会被当做上一个yield表达式的返回

// 执行顺序
function* test(x) {
  console.log(111)
  let a = yield x + 222

  console.log('a', a)
  let b = yield 333 + a

  console.log('b', b)
  let c = yield 44444 + b

  console.log('c', c)
}


let t = test(1)

console.log(t.next())
//111
// { value: 223, done: false }

// next 方法的参数
// yield表达式本身没有返回值,或者说总是返回undefined。
// next方法可以带一个参数,该参数就会被当作上一个yield表达式的返回值。
console.log(t.next(33))
// a 33
// { value: 366, done: false }

console.log(t.next('cc'))
//b cc
// { value: '44444cc', done: false }

console.log(t.next('dd'))
// c dd
// { value: undefined, done: true }

generator函数的运行顺序如下:

image-20201023154710301

generator搭配promise将异步回调变成同步模式

function request(data) {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      resolve(it.next(data + 1))
    }, 1000)
  })
}

function* getApi() {
  let res1 = yield request(1)
  console.log('res1', res1)
  let res2 = yield request(res1)
  console.log('res2', res2)
  let res3 = yield request(res2)
  console.log('res3', res3)
}

let it = getApi()
it.next()

// res1 2
// res2 3
// res3 4
上一篇下一篇

猜你喜欢

热点阅读