ES6生成器怎么使用

发布时间:2020-10-10 17:50:32 作者:小新
来源:亿速云 阅读:141

小编给大家分享一下ES6生成器怎么使用,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!

概念

生成器是由生成器函数( generator function )运行后得到的,是可迭代的。

function* gen() { 
  yield 'a';
  yield 'b';
  yield 'c';
}

let g = gen(); 
// "Generator { }"

原理及简单运用

生成器有一个很大的特点,它可以暂停内部代码运行,返回一个值给外部函数。(暂停后不会阻止其他代码运行)当外部调用其 next 方法后,会继续执行剩下的代码,并接受外部传来的一个参数。这个实现主要依赖于关键字 yield 。

yield 关键字使生成器函数执行暂停,yield 关键字后面的表达式的值返回给生成器的调用者。它可以被认为是一个基于生成器的版本的 return 关键字。

function* g(){
    var a = yield 2;
    console.log(a);
}

var it = g(); // 返回一个可迭代的生成器对象
console.log(it.next()); // 执行生成器函数内部代码,第一次返回 {done: false, value: 2}
it.next(3); // 继续执行生成器函数内部代码,同时向生成器传递参数3,最后返回 {done: true, value: undefined}

一个简单的计数器

function* count(){
    var n = 1;
    while(true){
        yield n++;
    }
}

var it = count();
it.next(); // 1
it.next(); // 2
it.next(); // 3

用同步方式写异步代码

以前处理异步 ajax 请求结果,一般采用传递回调函数的方式。一旦遇到多层回调嵌套,代码的可读性会降低,并且调试起来也不方便。有了生成器之后,我们就可以用同步的方式写异步的代码。这听上去非常的有意思。我们的代码将会是这样的

function foo(){
    var result = asyncFun(); // asyncFun 是异步函数,result 是异步返回的结果
    console.log(result);
}

当然,上面的代码并不能得到正确的结果,它只是一个设想。我们正打算用生成器来实现,而且这是可行的。想想生成器有哪些特点:

  1. 它能暂停,向外部返回值
  2. 能继续执行剩下的代码,并接受外部传来的参数

这就足够了。有了生成器函数,现在我们重新来设计代码:

function* foo(){
    // 这里遇到了异步方法,必须停下来。
    // 等待异步方法执行完毕,并返回结果,继续运行代码。当然,同步 ajax 不能算,它不是异步
    // 输出结果
}

静下来想一想有哪些关键字,与暂停、继续有关。停下来...继续...停下来...继续...停下来...继续...Don't...Stop...Don't...Stop...Don't...Stop......这两个词就是 yieldnext.

function *foo(){
    var result = yield asyncFun(next);
    console.log(result);
}

当代码遇到 yield 会暂停,这个时候 asyncFun 函数是不会暂停的,会执行,等执行完毕,再调用生成器的 next 方法,并将返回结果作为参数传给 next。由于在生成器函数内部我们拿不到 next,必须借助于全局变量来传递 next

var next, gn;

function asyncFun(next){
    // 模拟异步请求
    setTimeout(function(){
        // 返回一个随机数
        next(Math.random())
    }, 1000)
}

function* foo(){
    var result = yield asyncFun(next);
    console.log(result);
}

gn = foo();
next = gn.next.bind(gn);
next(); // 打印随机数

这样写,运行看上去有些繁重。可以写一个包装函数运行含有异步代码的生成器函数。

function asyncFun(next){
  // 模拟异步请求
  setTimeout(function(){
    // 返回一个随机数
    next(Math.random())
  }, 1000)
}

function* foo(){
  var result = yield function(next){asyncFun(next)};
  console.log(result);
}



function wrapFun (gFn){
  var gn = foo(),
      next = gn.next.bind(gn);
  next().value(next);
}

wrapFun(foo);

演示地址

不过,自从出了 Promiseawait 之后,更多的是用这个组合,其使用也更简单,范围也更广。

以上是ES6生成器怎么使用的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注亿速云行业资讯频道!

推荐阅读:
  1. ES6中生成器函数的示例分析
  2. 如何使用Python生成器

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

javascript

上一篇:javascript中reduce如何使用

下一篇:innerhtml的语法是什么

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》