广告
返回顶部
首页 > 资讯 > 前端开发 > JavaScript >JavaScript迭代器与生成器使用详解
  • 802
分享到

JavaScript迭代器与生成器使用详解

摘要

目录迭代器 (Iterator)Iterator工作原理自定义遍历数据生成器 (Generator)生成器参数传递使用生成器实现回调地狱功能生成器函数实例生成器—thro

迭代器 (Iterator)

迭代器(Iterator)也叫遍历器,是一种接口,为各种不同的数据结构提供统一的访问机制。任何数据结构只要部署 Iterator 接口,就可以完成遍历操作;js中原有表示“集合”的数据结构,主要是数组(Array)和对象(Object),es6又新增了 Map 和 Set,这样就有了四种数据集合。

如果用户组合使用四种不同的数据结构,比如数组的成员是对象或者对象的成员是Map,这样就需要一种统一的接口机制,来处理所有不同的数据结构,这里就需要借助 Iterator ,其作用为:为各种数据结构提供统一简便的访问接口、使数据结构的成员能够按某种次序排列、给ES6新增的遍历方法 for...of 提供消费。

Iterator工作原理

?‍需要自定义遍历数据的时候,要想到迭代器,以下是使用原理:

创建一个指针对象,指向当前数据结构的起始位置

第一次调用对象的 next 方法,指针自动指向数据结构的第一个成员

接下来不断调用 next 方法,指针一直往后移动,直到指向最好一个成员

?每调用 next 方法返回一个包含 value 和 done 属性的对象

<script>
    // 声明一个数组
    const animals = ['大象','狮子','老虎','猎豹','猴子']
    // 创建一个指针对象
    let iterator = animals[Symbol.iterator]()
    // 调用对象的next方法
    console.log(iterator.next());
    console.log(iterator.next());
    console.log(iterator.next());
    console.log(iterator.next());
    console.log(iterator.next());
    console.log(iterator.next());
</script>

next()方法:返回一个对象,表示当前数据成员的信息。这个对象具有 value 和 done 两个属性,value属性返回当前位置的成员,done属性是一个布尔值,表示遍历是否结束,即是否还有必要再一次调用next()方法。

自定义遍历数据

当我们使用 for...of 循环遍历某种数据结构时,该循环会自动去寻找 Iterator 接口,其接口默认部署在 Symbol.iterator 属性上,Symbol.iterator属性本身就是一个函数,就是当前数据结构默认遍历器生成函数,执行这个函数就会返回一个遍历器。

<script>
    // 声明一个对象
    const classroom = {
        name:'终极一班',
        team: [
            '汪大东',
            '金宝三',
            '花灵龙',
            '中万钧',
            '雷婷'
        ],
        [Symbol.iterator](){
            // 索引变量
            let index = 0
            // 引入函数外的this
            let _this = this
            return {
                next:function(){
                    if(index<_this.team.length){
                        const result = {value:_this.team[index],done:false}
                        // 下标自增
                        index++
                        // 返回结果
                        return result
                    }else{
                        return {value:undefined,done:true}
                    }
                }
                // 对于遍历器对象来说,done: false和value: undefined属性都是可以省略的,因此上面的makeIterator函数可以简写成下面的形式
                // next:function(){
                //     return index<_this.team.length ? {value:_this.team[index++]}:{done:true}
                // }
            }
        }
    }
    // 遍历这个对象
    for(let con of classroom){
        console.log(con);
    }
</script>

生成器 (Generator)

生成器函数是ES6提供的一种异步操作编程方案,语法行为与传统函数完全不同。以前我们进行异步编程的方法就是纯回调函数,Generator函数与普通函数的区别在于:function关键字与函数名之间有一个星号,函数体内部使用yield表达式,具体案例如下:

<script>
    // 生成器其实就是一个特殊的函数,在function与函数名中间补上一个星号
    // yield就是一个函数分隔符,使生成器函数执行暂停,,yield关键字后面的表达式的值返回给生成器的调用者
    function * person() {
        console.log('hello world');
        yield '第一分隔线' 
        console.log('hello world 1');
        yield '第二分隔线'
        console.log('hello world 2');
        yield '第三分隔线'
    }
    let iterator = person()
    // console.log(iterator); 打印的就是一个迭代器对象,里面有一个 next() 方法,我们借助next方法让它运行
    iterator.next()
    iterator.next()
    iterator.next()
</script>

既然上文代码是一个迭代器对象,我们可以用 for...of 进行一个遍历。

<script>
    function * person() {
        yield '第一分隔线' 
        yield '第二分隔线'
        yield '第三分隔线'
    }
    for(let v of person()){
        console.log(v);//打印是yield后面字符串的内容
    }
    console.log('-----------------');
    // 方法补充:yield* 后面跟的是一个可遍历的结构,它会调用该结构的遍历器接口。
    let generator = function* () {
        yield 1;
        yield* [2,3,4];
        yield 5;
    };
    var iterator = generator();
    console.log(iterator.next());// { value: 1, done: false }
    console.log(iterator.next()); // { value: 2, done: false }
    console.log(iterator.next()); // { value: 3, done: false }
    console.log(iterator.next()); // { value: 4, done: false }
    console.log(iterator.next()); // { value: 5, done: false }
    console.log(iterator.next()); // { value: undefined, done: true }
</script>

生成器参数传递

生成器是可以进行参数传递的,传递的参数还是需要借助next方法才可以,next方法也是可以传递参数的,传递的参数是作为上一个 yield 的返回结果,说白了就是将原来的值给覆盖了。

<script>
    function * person(arg) {
        console.log(arg);
        let one = yield 111
        console.log(one);
        let two = yield 222
        console.log(two);
        let three = yield 333
        console.log(three);
    }
    // 执行获取迭代器对象
    let iterator = person('AAA')
    console.log(iterator.next());
    // next方法传入实参
    console.log(iterator.next('BBB'));
    console.log(iterator.next('CCC'));
    console.log(iterator.next('DDD'));
</script>

Genterator 函数从暂停状态到恢复运行,它的上下文状态(context)是不变的。通过next方法的参数,就有办法在 Generator 函数开始运行之后,继续向函数体内部注入值。也就是说,可以在 Generator 函数运行的不同阶段,从外部向内部注入不同的值,从而调整函数行为。

使用生成器实现回调地狱功能

ES6诞生之前,异步编程大致有四种:回调函数、事件监听、发布/订阅、Promise对象,如下案例讲解回调地狱的实现:

<script>
    // 案例: 1s打印111,2s打印222,3s打印333
    // 回调地狱
    setTimeout(()=>{
        console.log(111);
        setTimeout(()=>{
            console.log(222);
            setTimeout(()=>{
                console.log(333);
            },3000)
        },2000)
    },1000)
    // 生成器
    function one(){
        setTimeout(()=>{
            console.log(111);
            iterator.next()
        },1000)
    }
    function two(){
        setTimeout(()=>{
            console.log(222);
            iterator.next()
        },2000)
    }
    function three(){
        setTimeout(()=>{
            console.log(333);
            iterator.next()
        },3000)
    }
    function * gen(){
        yield one()
        yield two()
        yield three()
    }
    let iterator = gen()
    iterator.next()
</script>

生成器函数实例

生成函数在异步任务这一方面的表现

<script>
    // 案例:用户信息 商品信息 商品价格
    function getUsers (){
        setTimeout(()=>{
            let data = '用户信息'
            iterator.next(data)
        },1000)
    }
    function getGoods (){
        setTimeout(()=>{
            let data = '商品信息'
            iterator.next(data)
        },1000)
    }
    function getPrice (){
        setTimeout(()=>{
            let data = '商品价格'
            iterator.next(data)
        },1000)
    }
    // 生成器
    function * gen(){
        let users = yield getUsers()
        console.log(users);
        let goods = yield getGoods()
        console.log(goods);
        let price = yield getPrice()
        console.log(price);
    }
    // 调用生产器函数
    let iterator = gen()
    iterator.next()
</script>

生成器—throw()

Generator 函数返回的遍历器对象,都有一个throw方法,可以在函数体外抛出错误,然后在 Generator 函数体内捕获。如果生产器函数内部没有部署try...catch代码块,那么抛出的错误会直接被外部的catch代码块捕获。如果 Generator 函数内部和外部,都没有部署try...catch代码块,那么程序将报错,直接中断执行。

<script>
    var g = function* () {
    try {
        yield;
    } catch (e) {
        console.log('内部捕获', e);//a
    }
    };
    var i = g();
    i.next();
    try {
        i.throw('a');
        i.throw('b');
    } catch (e) {
        console.log('外部捕获', e);//b
    }
</script>

生成器—return()

生成器的return方法,可以返回给定的值,并且终结遍历 Generator 函数。

<script>
    function* gen() {
        yield 1;
        yield 2;
        yield 3;
    }
    var g = gen();
    console.log(g.next())        // { value: 1, done: false }
    // 遍历器对象g调用return()方法后,返回值的value属性就是return()方法的参数foo。并且,Generator 函数的遍历就终止了
    console.log(g.return('foo')) // { value: "foo", done: true }
    console.log(g.next())        // { value: undefined, done: true }
    // 如果return()方法调用时,不提供参数,则返回值的value属性为undefined。
    function* gen1() {
        yield 1;
        yield 2;
        yield 3;
    }   
    var g1 = gen1();
    console.log(g1.next()) // { value: 1, done: false }
    console.log(g1.return()) // { value: undefined, done: true }
</script>

如果 Generator 函数内部有try...finally代码块,且正在执行try代码块,那么return()方法会导致立刻进入finally代码块,执行完以后,整个函数才会结束,调用return()方法后,就开始执行finally代码块,不执行try里面剩下的代码了,然后等到finally代码块执行完,再返回return()方法指定的返回值

<script>
    function* numbers () {
        yield 1;
        try {
            yield 2;
            yield 3;
        } finally {
            yield 4;
            yield 5;
        }
        yield 6;
    }
    var g = numbers();
    console.log(g.next()) // { value: 1, done: false }
    console.log(g.next()) // { value: 2, done: false }
    console.log(g.return(7)) // { value: 4, done: false }
    console.log(g.next()) // { value: 5, done: false }
    console.log(g.next()) // { value: 7, done: true }
</script>

生成器简写

如果一个对象的属性是 Generator 函数,可以用如下形式进行简写:

<script>
    let obj = {
        // myGeneratORMethod: function * (){}
        // 上面代码可以简写成如下形式,在属性前加一个 Generator 函数即可。
        * myGeneratorMethod(){}
    }
</script>

到此这篇关于javascript迭代器与生成器使用详解的文章就介绍到这了,更多相关JS迭代器与生成器内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: JavaScript迭代器与生成器使用详解

本文链接: https://www.lsjlt.com/news/171007.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • JavaScript迭代器与生成器使用详解
    目录迭代器 (Iterator)Iterator工作原理自定义遍历数据生成器 (Generator)生成器参数传递使用生成器实现回调地狱功能生成器函数实例生成器—thro...
    99+
    2022-11-13
    JavaScript迭代器与生成器 JavaScript生成器 JavaScript迭代器
  • python迭代器与生成器详解
    例子 老规矩,先上一个代码: def add(s, x): return s + x def gen(): for i in range(4): yield i base = gen() ...
    99+
    2022-06-04
    生成器 详解 迭代
  • Python 迭代器与生成器实例详解
    Python 迭代器与生成器实例详解 一、如何实现可迭代对象和迭代器对象 1.由可迭代对象得到迭代器对象 例如l就是可迭代对象,iter(l)是迭代器对象 In [1]: l = [1,2,3,4] ...
    99+
    2022-06-04
    生成器 详解 实例
  • Python生成器与迭代器详情
    目录1、生成器 2、迭代器与可迭代的生成器1、生成器 现在可以通过生成器来直接创建一个列表,但是由于内存的限制,列表的容量肯定是有限的,如果我们需要一个包含几百个元素的列表,但是每...
    99+
    2022-11-12
  • python迭代器,生成器详解
    目录迭代器 生成器 总结迭代器 聊迭代器前我们要先清楚迭代的概念:通常来讲从一个对象中依次取出数据,这个过程叫做遍历,这个手段称为迭代(重复执行某一段代码块,并将每一次迭代得到的结...
    99+
    2022-11-12
  • Python 中迭代器与生成器实例详解
    Python 中迭代器与生成器实例详解 本文通过针对不同应用场景及其解决方案的方式,总结了Python中迭代器与生成器的一些相关知识,具体如下: 1.手动遍历迭代器 应用场景:想遍历一个可迭代对象中的所...
    99+
    2022-06-04
    生成器 详解 实例
  • 迭代器与生成器
    这一部分待加强!                (一)迭代器    一:简介    迭代是Python最强大的功能之一,是访问集合元素的一种方式。    迭代器是一个可以记住遍历的位置的对象。    迭代器对象从集合的第一个元素开始访问,...
    99+
    2023-01-30
    生成器 迭代
  • 详解Python中迭代器和生成器的原理与使用
    目录1.可迭代对象、迭代器1.1概念简介1.2可迭代对象1.3迭代器1.4区分可迭代对象和迭代器1.5可迭代对象和迭代器的关系1.6可迭代对象和迭代器的工作机制1.7自己动手创建可迭...
    99+
    2022-11-11
  • Python元类与迭代器生成器案例详解
    1.__getattr__和__getattribute__魔法函数 __getattr__是当类调用一个不存在的属性时才会调用getattr魔法函数,他传入的值item就是你这个调...
    99+
    2022-11-12
  • JavaScript中的迭代器和可迭代对象与生成器
    目录1. 什么是迭代器?1.1 迭代器的基本实现1.2 迭代器的封装实现2. 什么是可迭代对象2.1 原生可迭代对象(JS内置)2.1.1 部分for of 演示2.1.2 查看内置...
    99+
    2022-11-13
  • python中的迭代器,生成器与装饰器详解
    目录迭代器生成器装饰器总结迭代器 每一个可迭代类内部都要实现__iter__()方法,返回一个迭代类对象,迭代类对象则定义了这个可迭代类如何迭代。 for循环调用list本质上是是调...
    99+
    2022-11-13
  • Python迭代器与生成器
    生成器仅仅拥有生成某种东西的能力,如果不用__next__方法是获取不到值得。创建一个生成器函数>>> def scq(): ...    print("11") # 当函数代码块中遇到yield关键字的时候,这个函数就是...
    99+
    2023-01-31
    生成器 迭代 Python
  • Python3 迭代器与生成器
    Edit笔记内容:Python3 迭代器与生成器 笔记日期:2017-10-28迭代器迭代是Python最强大的功能之一,是访问集合元素的一种方式。 迭代器是一个可以记住遍历的位置的对象。 迭代器对象从集合的第一个元素开始访问,直到所有的元...
    99+
    2023-01-31
    生成器 迭代
  • JavaScript前端迭代器Iterator与生成器Generator怎么使用
    这篇文章主要介绍“JavaScript前端迭代器Iterator与生成器Generator怎么使用”,在日常操作中,相信很多人在JavaScript前端迭代器Iterator与生成器Generator怎么使...
    99+
    2022-10-19
  • 详解Python之可迭代对象,迭代器和生成器
    目录一、概念描述二、序列的可迭代性三、经典的迭代器模式四、生成器也是迭代器五、实现惰性迭代器六、使用生成器表达式简化惰性迭代器总结 一、概念描述 可迭代对象就是可以迭代的对象,我们可...
    99+
    2022-11-12
  • 正确理解python迭代器与生成器
    目录一、迭代器二、生成器三、生成器函数3.1、zip(可迭代对象1,可迭代对象2......)3.2、enumerate(iterable[,start])一、迭代器 迭代器就是iter(可迭代对象函数)返回的对象,...
    99+
    2022-06-02
    python 迭代器 python 生成器
  • 详解ES6 中的迭代器和生成器
    目录1.迭代器2.生成器1.迭代器 Iterator是 ES6 引入的一种新的遍历机制。两个核心 迭代器是一个统一的接口,它的作用是使各种数据结构可以被便捷的访问,它是通过一个键为S...
    99+
    2022-11-13
    ES6 中的迭代器和生成器 ES6 迭代器 ES6生成器
  • Python中迭代器与生成器怎么使用
    一、迭代器(foreach)1、可迭代的对象内置有__iter__方法的都叫可迭代的对象。Python内置str、list、tuple、dict、set、file都是可迭代对象。x = 1.__iter__ # SyntaxError: ...
    99+
    2023-05-22
    Python
  • Python生成器与迭代器怎么用
    这篇文章给大家分享的是有关Python生成器与迭代器怎么用的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。1、生成器现在可以通过生成器来直接创建一个列表,但是由于内存的限制,列表的容量肯定是有限的,如果我们需要一个...
    99+
    2023-06-25
  • python迭代器与生成器小结
    2016.3.10关于例子解释的补充更新 源自我的博客 例子 老规矩,先上一个代码: def add(s, x): return s + x def gen(): for i in range(4): ...
    99+
    2023-01-31
    生成器 小结 迭代
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作