我的编程空间,编程开发者的网络收藏夹
学习永远不晚

JS前端使用迭代器和生成器原理是什么

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

JS前端使用迭代器和生成器原理是什么

这篇文章主要介绍了JS前端使用迭代器和生成器原理是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JS前端使用迭代器和生成器原理是什么文章都会有所收获,下面我们一起来看看吧。

正文

生成器和迭代器这两个东西平时作为一个切图仔,一直都没有使用到。感觉是只有在面试之前才会的东西。面试过不了几天,再次看这两个词一阵恍惚。

记忆力退化成这样了么?最大的原因一定是用得少了。然后呢?就是没有真正的理解它们。我对于它们的认知常常有下面这些:

我常常把迭代器和生成器理解成完全不同的东西。

我常常把for、forEach、map、reducefor of混为一谈

我常常把数组、类数组认为是可迭代对象

想来要真正的记住它,增加自己的武器库,必须要弄明白这些东西才行。

我们首先是要搞明白什么for of是干什么用的。

业务代码确实使用不上,但是如果不理解的话,等真到了可以使用的场景的时候,又是否真的能够运用起来,甚至记起来呢?

for of 是干什么用的

所有人都知道一些概念for、forEach、map、reduce这些是可以遍历数组的,for of是用于遍历迭代对象的。如下:

const arr = [1, 2, 3]arr.forEach((item, index) => {   console.log(item) // 1, 2, 3    console.log(index) // 0, 1, 2})

而巧合的是for of也可以遍历数组

for (let key of arr) {    console.log(key) // 1 2 3}

将arr改变为const obj = { a: 1, b: 2, c: 3 }的时候,两者都没有遍历出结果。

前者是没有反应,后者会直接报错:TypeError: obj is not iterable。翻译一下,类型错误:obj 不是一个可迭代对象。

那么什么是可迭代对象呢?

可迭代对象是什么?

我们先来看看下面这个例子:

const itemLi1 = document.getElementByTagName('li')const itemLi2 = document.querySelectorAll('li')for(let key of itemLi1) {    console.log(item)}for(let key of itemLi2) {    console.log(item)}

也就是说HTMLCollectionNodeList是可以迭代对象。其他的可迭代对象有Array、map、set、string等等。如果说类数组的话,是不是迭代对象呢?

const arrLike = {  0: 1,  1: 2,  2: 3,  lenght: 3}for (let i = 0; i < arrLike.length; i++) {    console.log(arrLike[i]) // 1, 2, 3}for (let of arrLike) {    console.log(key) // uncachh TypeError: obj is not iterable}

for循环打印出了对应的结果,而for of 报错了。类数组不是可迭代的的对象。这又是为什么呢?我们将类数组和HTMLCollection类型打印出来比较一下。

JS前端使用迭代器和生成器原理是什么

而类数组如下:

JS前端使用迭代器和生成器原理是什么

它们有一个明显的不同,可迭代对象的原型链上面是包括Symbol.iterator的。而这个就是让数组变成可迭代的根本原因。

也就是说,当目的对象的原型链上面包括Symbol.iterator的时候,它才是可迭代对象。

对象是无序的,无序的东西自然不可以迭代

这里使用到了Symbol类型,它在MDN上面的解释就是用于生成全局唯一的变量。而可迭代对象就是它的使用场景。受它的启发,我们在业务当中,如果需要前端来生成一个唯一ID的时候,再次之前,通常都是创建一个UUID的函数来生成一个唯一ID。Symbol不用这么麻烦,直接使用就可以了。

由此可知,Array.prototype[Symbol.iterater]这个函数封装了一些东西,使得for of可以将对象的元素给打印出来。

换一句话来说,就是Array.prototype[Symbol.iterater] = function() {}的执行生成一个迭代器对象。

也就是说,当Object.prototype也有[Symbol.iterater]的方法的时候,for of也能够遍历它呢?我们来试试看吧。

Object.ptotoype[Symbol.iterator] = function value() {}

这不就是生成器的作用么?

生成器和迭代器的关系。

ES6给我提高了一个生成器的函数。既然叫做生成器,它生成的东西就是迭代器。

表现形式如下:

function * generation(iterableObject) {    for(let i = 0; i < iterableObject; i++) {        yield iterableObject[i]    }}

*符号和yield关键字组成。

const iterator = generation([1, 2, 3]), 其执行流程如下:

iterator.next() ==> { value: 1, done: false }

iterator.next() ==> { value: 2, done: false }

iterator.next() ==> { value: 3, done: false }

iterator.next() ==> { value: undefined, done: true }

到了第四次,value为undefined的时候,done为true(也就是说,当done为true的时候,value一定为undefined)。所以说,yield的作用有两个:

  • 生成一个值,将该值封装成一个对象,而这个对象是{ value: .., done: flase/true }这样的形式。

  • 停下来

可以明显的看出来,生成器有一个作用,通过next这个接口,可以看到迭代的过程。

既然说生成器生成了一个迭代器,那么是不是说生成器执行后的结果就是一个迭代器呢?既然是迭代器,自然就可以被for of给遍历。

for (const key of generation([1, 2, 3]) {    console.log(key) // 1, 2, 3}

果然可以。

经典面试题: 自己实现一个next这样的接口呢?

上面已经有了实现的思路。通过一个标识符和一个判断就能够使用ES5来使用,如下代码片段。

function generation(iterableObj) {    let nextIndex = 0    function next() {}    return {        next: () => {            return nextIndex < iterableObj.length                             ? { value: iterableObj[nextIndex++], done: false }                             : { value: undefined, done: true }         }    }}

当nextIndex下于数组长度的时候,没有迭代完毕。

注意:nextIndex++是先跑nextIndex,再自增。

何为接口,后台给你一个url地址,这个是网络接口。next是设计师给你封装的一个方法,你通过这个方法来达到上吧yield的两个作用,所以next()也是一个接口,前端接口。简单来说,一个封装好的方法就是一个接口。

让非迭代对象也可以使用for of 进行遍历

正如第一节所说,Symbol.iterator的方法是迭代器的关键。那么我们也可以给Object挂载上该方法。既然该方法可以让对象变成迭代器,就可以直接使用上面ES5实现next方法的代码片段。

const obj = {  a: 1,  b: 2,  c: 3}Object.prototype[Symbol.iterator] = function value() {  const keys = Object.keys(Object(this))  let nextIndex = 0  function next() {    return nextIndex < keys.length        ? { value: [keys[nextIndex], obj[keys[nextIndex ++]]], done: false }        : { value: undefined, done: true }  }  return {    next  }}for (const [key, value] in obj) {  console.log(key)}

for循环和for in的关系

for循环和for in 看着很像,其实只是共用了for这个关键字,它们都是JS引擎底层实现的东西。和forEach、map这些是基于for循环的API不同,它们是在实现在for循环之上的。

关于“JS前端使用迭代器和生成器原理是什么”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“JS前端使用迭代器和生成器原理是什么”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

JS前端使用迭代器和生成器原理是什么

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

下载Word文档

猜你喜欢

JS前端使用迭代器和生成器原理是什么

这篇文章主要介绍了JS前端使用迭代器和生成器原理是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JS前端使用迭代器和生成器原理是什么文章都会有所收获,下面我们一起来看看吧。正文生成器和迭代器这两个东西平时作
2023-07-05

详解JS前端使用迭代器和生成器原理及示例

这篇文章主要为大家介绍了详解JS前端使用迭代器和生成器原理及示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2023-02-21

python迭代器和生成器的作用是什么

这篇文章主要讲解了“python迭代器和生成器的作用是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“python迭代器和生成器的作用是什么”吧!说明1、容器是可迭代对象,可迭代对象调用i
2023-06-20

Python中的迭代器和生成器的区别是什么?

Python中的迭代器和生成器的区别是什么?在Python编程中,迭代器(iterator)和生成器(generator)都是用于处理可迭代对象的工具。它们两者都可以用于遍历数据,但是在实现上却有一些不同之处。迭代器是一个对象,它实现了迭代
2023-10-22

Python迭代器的原理是什么及怎么使用

这篇文章主要介绍了Python迭代器的原理是什么及怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Python迭代器的原理是什么及怎么使用文章都会有所收获,下面我们一起来看看吧。什么是迭代器能被 next
2023-06-30

Python中的迭代和生成器的区别是什么?

Python中的迭代和生成器是两种不同的概念,它们在处理数据集合时有着不同的表现和用法。本文将详细介绍迭代和生成器的区别,并提供具体的代码示例。首先,让我们来了解一下迭代和生成器的概念。迭代是一种重复执行某一段代码的方法,它可以遍历一个序列
2023-10-22

Python中的迭代器和生成器的优劣势和适用场景是什么?

Python中的迭代器和生成器的优劣势和适用场景是什么?迭代器和生成器是Python中常用的编程概念,它们可以帮助我们更有效地处理大量数据,提高程序的性能和可读性。这篇文章将详细介绍迭代器和生成器的优劣势,并给出一些适用场景的具体代码示例。
2023-10-22

Python中的迭代器模式和生成器模式的适用场景是什么?

Python中的迭代器模式和生成器模式的适用场景是什么?迭代器模式和生成器模式是两种常用的设计模式,它们都用于处理集合(容器)中的元素,使得对集合的遍历更加简洁高效。下面将具体介绍这两种模式的适用场景,并提供相应的代码示例。迭代器模式是一种
2023-10-22

网络代理服务器无法使用的原因是什么

这篇文章主要介绍“网络代理服务器无法使用的原因是什么”,在日常操作中,相信很多人在网络代理服务器无法使用的原因是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”网络代理服务器无法使用的原因是什么”的疑惑有所
2023-06-25

Python中的装饰器和上下文管理器的原理和使用场景是什么?

Python中的装饰器和上下文管理器是两个非常有用的特性,它们可以帮助我们更好地组织和管理代码,并提高代码的可复用性。本文将分别介绍装饰器和上下文管理器的原理和使用场景,并给出具体的代码示例。一、装饰器的原理和使用场景原理:装饰器是一种在不
2023-10-22

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录