Javascript迭代器

举报
.Han 发表于 2021/08/28 15:03:54 2021/08/28
【摘要】 我们都知道可以用 for…of… 来循环可迭代对象,例如循环ArrayArgumentsMapSetStringTypedArrayNodeList 迭代器的工作原理创建一个指针,指向当前数据结构的起始位置第一次调用对象next方法,指针自动指向数据结构的第一个成员接下来不断调用next,指针一直往后移动,直到指向最后一个成员每调用一次next方法返回一个包含value和done属性的对象 ...

我们都知道可以用 for…of… 来循环可迭代对象,例如循环

Array
Arguments
Map
Set
String
TypedArray
NodeList

迭代器的工作原理

  • 创建一个指针,指向当前数据结构的起始位置
  • 第一次调用对象next方法,指针自动指向数据结构的第一个成员
  • 接下来不断调用next,指针一直往后移动,直到指向最后一个成员
  • 每调用一次next方法返回一个包含value和done属性的对象

为什么Object不可以被迭代

let arr = [1, 2, 3]
let obj = {
    a: 1,
    b: 2,
    c: 3
}
for(let val of arr) {
    console.log(val)
}
for(let val of obj) {
    console.log(val)
}

运行可以看到,数组正常循环没有问题,但是到了obj,会报:obj is not iterable(obj不是一个可迭代对象、obj不可被迭代)
这是为什么呢?将arr和obj分别使用 console.dir() 进行打印查看

可以看到,arr的原型上挂载着一个 Symbol[Symbol.iterator],而obj的原型上并没有这个属性,因此这就是为什么obj不可以被迭代的原因
但是如果非得让obj也成为一个 可迭代对象,如何去实现呢?此时就需要给obj身上加上一个迭代器

let obj = {
    a: 1,
    b: 2,
    c: 3,
    [Symbol.iterator]: function () {
        let values = Object.values(obj)
        let index = 0
        /* 必须得return 一个对象出去 */
        return {
            /* 这是固定写法 */
            next() {
                if (index >= values.length) {
                    return {
                        done: true // 此时迭代结束,就不需要再return value出去
                    }
                } else {
                    return {
                        value: values[index++],
                        done: false
                    }
                }
            }
        }
    }
}
for (let item of obj) {
    console.log(item)
}

运行上方的代码,此时obj已经可以使用 for…of… 进行循环,并且此时的obj已经是一个可迭代对象

转自 Javascript迭代器 - 韩小韩博客

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。