JavaScript – Iterator
参考
前言
es6 以后经常可以看到 for...of 的踪迹. 如果你细看会发现它挺神奇的. 不只是 Array 可以被 for...of, 有些对象也可以被 for...of.
const array = [1, 2, 3];
console.log(Array.isArray(array)); // true
for (const value of array) {
console.log(value); // 1, 2, 3
}
上面 Array 能被 for...of
下面 isArray 是 false, querySelectorAll 返回的是一个 NodeListOf 对象. 但是依然可以被 for...of.
const notArray = document.querySelectorAll('h1');
console.log(Array.isArray(notArray)); // false
for (const value of notArray) {
console.log(value); // h1
}
其原因是, 只要是 Iterable 对象就可以被 for...of 遍历. 而 Array 和 NodeListOf 都属于 Iterable 对象, 所以它们都可以被遍历.
Iterable 对象
这是 TypeScript 对 Iterable 对象的描述
interface Iterable<T> {
[Symbol.Iterator](): Iterator<T>;
}
一个对象里有一个 Symbol.Iterator 方法, 就是一个 Iterable 对象. 之前在 JavaScript – Symbol 里, 我有提到了 es6 如何利用各种 build-in Symbol 来做许多事情, 其中一个就是 Iterator 了.
不只 JS, C# 也是有 Iterator 的概念. List<string> 的 List 就是一个 Iterable 对象来的, 所以它支持 foreach.
Iterator
Iterable 对象的 Symbol.Iterator 方法必须返回一个 iterator.

这个 Iterator 也是一个对象, 它有一个最重要的方法 next, 下面我们来看看 Iterator 对象怎么用.
How to use a Iterator
const Iterator: Iterator<string> = {
next() {
return { value: '1', done: true };
},
};
const { value, done } = iterator.next();
console.log(value); // '1'
console.log(done); // true
虽然我用 TypeScript 写, 但是你可以把它当 JavaScript 看, 去掉你看不懂的地方就可以了. 不影响理解的.
next 方法返回一个 value 和一个 boolean done 表示遍历是否完结了.
站在一个使用 Iterator 的角度, 我们可以了解到.
1. Iterator 是对象, 有一个 next 方法
2. 调用 next 可以拿到一个值, 同时可以知道是否遍历结束. 如果还没有结束意味着可以继续 next 拿到下一个值.
3. Iterator 内部如何实现遍历, 调用者是不知道的, 可能内部是一个 array 或者是一个 formula. 消费者不知道, 也不需要知道. 只要 call next 就对了.
for...of
了解了如何使用 Iterator, 就能理解 for...of 的运作逻辑了.
它只是封装了调用 .next 和判断 done 是否结束.
看看这个例子
const stringList: Iterable<string> & { value: string } = {
value: 'Derrick',
[Symbol.Iterator]() {
let index = 0;
return {
next: () => ({
value: this.value[index++],
done: index === this.value.length + 1,
}),
};
},
};
for (const value of stringList) {
console.log('value', value); // d e r r i c k
}
stringList.value = 'super';
for (const value of stringList) {
console.log('value', value); // s u p e r
}
stringList 是一个 Iterable, 一个拥有 Symbol.Iterator 方法的对象
Symbol.Iterator 返回一个 Iterator, 一个拥有 next 方法的对象. next 返回一个值和 done (表示是否遍历结束)
for...of 开始时, 会调用 Symbol.Iterator 方法得到 stringList 的 Iterator
接着调用 next 获取值放入 value, 这时 console 就会发出每一个 char 值. 这个逻辑写在 Iterator 内. for...of 并不知道 value 的由来, 它只是消费.
当 for...of 发现 done = true 时, 遍历就结束了.
Break and Return
在 for..of 内使用 break 会中断遍历, Iterator 的 return 方法会被调用.

Iterator 除了 next 方法, 还可以实现 return 和 throw 方法. throw 是配合 Generator 函数才会用到, 这篇先不提.
return 的用意是, 当调用者中断遍历时通知 iterator. 为了让 Iterator 内部做一些释放内存之类的事情.
所以一般的流程是 Iterable[Symbol.Iterator]() 获得 Iterator, 然后 .next 一直到结束
中断的情况则是调用了 next, 但在还没有 done 之前调用了 return.
for...of 内的 break 语句就会调用 iterator.return (注意: 只有 break 才是中断遍历, continue 并不是哦)
注: iterator.return 是可以传参数和返回值的, 但这个用于 Generator 函数. 一般 for...of 遍历不需要.
const stringList: Iterable<string> & { value: string } = {
value: 'Derrick',
[Symbol.Iterator]() {
let index = 0;
return {
next: () => ({
value: this.value[index++],
done: index === this.value.length + 1,
}),
return: (value) => {
console.log('return', value);
return {
value: '',
done: true,
};
},
};
},
};
for (const value of stringList) {
if (value === 'i') {
console.log('calling break');
break;
}
console.log('value', value); // d e r r i calling break return
}
Iterable + Spread Operator
Iterable > Iterator > next > values
这样步骤太多了, 有一个非常快捷的方法可以把所以 values 遍历放入 Array 中.
那就是配上 Spread Operator
const values = [...stringList]; // ['D', 'e', 'r', 'r', 'i', 'c', 'k']
JavaScript – Iterator的更多相关文章
- Javascript Iterator
[Javascript Iterator] 1.@@iterator Whenever an object needs to be iterated (such as at the beginning ...
- javascript里面支持el表达式和<s:iterator>
javascript不支持jstl标签,支持<s:iterator>和el表达式
- JavaScript 設計模型 - Iterator
Iterator Pattern是一個很重要也很簡單的Pattern:迭代器!我們可以提供一個統一入口的迭代器,Client只需要知道有哪些方法,或是有哪些Concrete Iterator,並不需要 ...
- javascript设计模式-迭代器模式(Iterator)
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- 【前端】【javascript】es6中的遍历器接口Iterator
好久没发文章啦-.-为了证明我还活着,我决定从笔记里面抓一篇还算不乱比较像文章的发出来... 这些笔记是我在学es6的时候断断续续记录的,最近会一份一份整理陆陆续续发出来,顺便也自己再看一遍.我学习e ...
- [Javascript] Creating an Iterator from an Array
Every Array has a function which you can use to create an iterator. This function can only be access ...
- JavaScript 对象部署 Iterator 接口
const name = { first:"hello", last:"world", fullname: "hello world" } ...
- 《你不知道的JavaScript》整理(三)——对象
一.语法 两种形式定义:文字形式和构造形式. //文字形式 var myObj = { key: value }; //构造形式 var myObj = new Object(); myObj.key ...
- 【转】浅谈JavaScript、ES5、ES6
什么是JavaScript JavaScript一种动态类型.弱类型.基于原型的客户端脚本语言,用来给HTML网页增加动态功能.(好吧,概念什么最讨厌了) 动态: 在运行时确定数据类型.变量使用之前不 ...
- 用vue.js学习es6(六):Iterator和for...of循环
一.Iterator (遍历器)的概念: 遍历器(Iterator)就是这样一种机制.它是一种接口,为各种不同的数据结构提供统一的访问机制.任何数据结构只 要部署Iterator接口,就可以完成遍历操 ...
随机推荐
- oeasy教您玩转vim - 89 - # 高亮细节Highlight
高亮细节 highight 回忆 这个自动命令 autocmd 还是很方便的 打开时.保存时就会有自动执行的操作 自动命令有这么几大元素 {event} 触发事件 {pattern} 文件模式 { ...
- 机器学习:详解迁移学习(Transfer learning)
详解迁移学习 深度学习中,最强大的理念之一就是,有的时候神经网络可以从一个任务中习得知识,并将这些知识应用到另一个独立的任务中.所以例如,也许已经训练好一个神经网络,能够识别像猫这样的对象,然后使用那 ...
- Divide Interval 题解
背景 太逊了,调了三次才调出来,所以写篇题解寄念.LC好睿智 题意 给你两个数 \(a,b\),现在要从 \(a\) 跑到 \(b\),每次可以将当前的 \(a\) 拆分成 \(2^n\times m ...
- 小技巧:初始化后查看容器内某一bean的信息
1.debug 2. 3.与容器名对应,可以看到容器的对应信息 4.输入表达式可以直接获取对应结果信息,这里查看的是默认SpringSecurity过滤链的bean
- Jmeter参数化3-正则表达式提取
如果你想要将jmeter上个接口返回的参数值传到下个接口请求参去,可用正则表达式来提取传递 1.首先得到jmeter接口的json返回结果 2.组件路径:线程组->右键添加->后置处理器- ...
- web3 产品介绍 Dune Analytics 区块链的数据探索和可视化 链上热点和趋势一手掌握
Dune Analytics 是一个强大的数据分析平台,旨在帮助用户在区块链上进行数据探索和可视化. Dune Analytics的特点: 数据查询与可视化:Dune Analytics允许用户从多个 ...
- 多智能体路径规划问题 —— Learn all about Multi-Agent Path Finding (MAPF)
地址: http://mapf.info/index.php/Main/Publications 与其对应的实验室地址: https://www.movingai.com/
- 为什么模具设计人员中很多人不喜欢使用CAE仿真软件呢? —— 以汽车制造领域为例
参考论文: 通常 的有限元分析软件在后处理 上只是将计算结果 以图形或表格的 方式显 示 在屏幕上 , 并没有对计算结果进行系统的整理 和提炼 . 在基于有 限元 计算结果 的反复设计过程 中, 设计 ...
- Google主打的机器学习计算框架——jax的升级包
相关: 机器学习洞察 | 一文带你"讲透" JAX Jax的主要应用场景: 深度学习 (Deep Learning):JAX 在深度学习场景下应用很广泛,很多团队基于 JAX 开发 ...
- 在vscode中通过修改launch.json文件为项目设置当前工作目录cwd——在launch.json文件中修改cwd变量
关于当前工作目录是什么以及其与模块搜索路径的区别可以参见下文: Python语言中当前工作目录(Current Working Directory, cwd)与模块搜索第一路径都是指什么??? --- ...