JS数组遍历的细节以及实现记录

本篇文章和大家了解一下JS数组遍历的细节以及实现记录。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。

目录
  • 1. 数组空位问题

    • 1.1 空位判断

    • 1.2 刚列举了数组的一些操作会对空位进行跳过,但其实,它们在处理上也还是存在一些差异的

  • 2. 是否修改原数组

    • 3. 附一下数组遍历的几种方法

      • 3.1 索引访问

      • 3.2 for ... of

      • 3.3 forEach

      • 3.4 for ... in

      • 3.5 map、filter、reduce

    1. 数组空位问题

    数组空位是什么,它是数组内某个元素没有任何值,这种现象称为空位现象,我们在使用Array()去构造一个数组时,如果只传入一个参数,数组里面的元素项就会出现空位现象,它其实是没有任何元素的,下面是几个会创建空位的典型例子:

    let arr = Array(5) // <5 empty items>
    let res = arr.map(e => e + 1) // <5 empty items>
    let a = [,,]; // <2 empty items>

    ES5 中的数组方法包括:map、filter、forEach、reduce,还有 for ... in 等语法,这些方法执行的时候遇到空位,会直接跳过。

    // ES5 及以前,不会处理空位
    const a = Array(5);
    console.log(a) // [ <5 empty items> ]
    console.log(a.map(_ => 1)) // 无效,[ <5 empty items> ]
    // console.log(a.reduce((p, c)=> p+c)) // 报错
    console.log(a.filter(x => true)) // []
    a.forEach(e => {
      console.log(e);
    }) // 无任何输出
    console.log(a+'') // ,,,,
    console.log(a.indexOf(undefined)) // -1
    console.log(a.lastIndexOf(undefined)) // -1
    for (let k in a) {
      console.log(k) // 无任何输出
    }

    ES5中将空位视为undefined的有:join(),toString()

    [,'a','b',undefined,null].join("@") // '@a@b@@'
    [,'a','b',undefined,null].toString() // ',a,b,,'

    ES6,一些操作会将数组空位视为undefined来处理。比如下面的一些操作

    console.log([...a]) // [ undefined x 5 ]
    console.log(Array.from(a)) // [ undefined x 5 ]
    console.log(a.includes(undefined)) // true
    console.log(a.find(x => x === undefined)) // undefined
    console.log(a.findIndex(x => x === undefined)) // 0
    // for of 循环取值
    for (let k of a) {
      console.log(k) // 输出5个undefined
    }

    1.1 空位判断

    我们可以使用in运算符来判断数组某个位置是否存在空位

    let arr = Array(3) console.log(0 in arr) // false

    如上述所示,使用in运算符,判断索引为0的位置是否为空位,遇到数组空位会返回false
    这里需要注意的是所谓的空位是没有任何值的,undefined、null它们都不属于空位,下面来判断一下。

    let arr = [undefined, null];
    console.log(0 in arr) // true
    console.log(1 in arr) // true

    验证了undefined、null所在的位置它不是空位

    1.2 刚列举了数组的一些操作会对空位进行跳过,但其实,它们在处理上也还是存在一些差异的

    比如:forEach、filter、some与every等在遇到空位时,会直接跳过它,不会保留它的值, 而map则会保留空位
    我们来看下例子:

    [,1,2].forEach((x,i) => { console.log(i); }) // 这里只会输出1 2
    [,1,2].filter((x,i) => { return x > 0; }) // 这里也只会输出[1, 2]
    [,1,2].map((x) => { return x > 0; }) // 这里会输出[空, true, true],它保留了空位

    其余的可以自行测试看看结果

    2. 是否修改原数组

    请看以下代码,遍历过程中,试图修改遍历的每一项。

    const arr = [1, 2, 3]
    let res = arr.filter(item => {
      item++
      return item >= 3
    })
    console.log(arr) // [ 1, 2, 3 ]
    console.log(res) // [ 2, 3 ]

    filter 里的 item 自增了,所以最后有两项符合过滤规则,但是原数组并没有变。这说明这里的 item 只是原数组项的一个值拷贝,数组遍历是按值传递的。
    再看一段代码,这次我们遍历引用数据类型的数组。

    const brr = [
      {count: 1},
      {count: 1},
      {count: 1}
    ]
    const res2 = brr.map(item => {
      item.count++
      return item.count
    })
    console.log(brr) // [ { count: 2 }, { count: 2 }, { count: 2 } ]
    console.log(res2) // [ 2, 2, 2 ]

    这里的修改改变了原数组,那之前说的按值传递有问题?其实还没问题的,数组函数方法说到底也是一个函数,JS 的函数总是按值传递的:

    基本数据类型(包含变量、函数参数等)存储在栈(stack)中,传递参数会复制一份值
    引用数据类型的引用(指针)存储在栈中,指向的值存储在堆(heap)中,传递参数会复制一份对象的引用地址,复制的引用地址和原引用地址指向堆中同一个对象(因此修改参数,也修改了原对象)

    3. 附一下数组遍历的几种方法

    3.1 索引访问

    const arr = [1, 2, 3]
    for (let i = 0; i < arr.length; i++) {
      console.log(arr[i])
    }

    3.2 for ... of

    const arr = [1, 2, 3]
    for (let e of arr) {
      console.log(e)
    }

    3.3 forEach

    注意:return 和 break 无法中断遍历。

    • return 可以跳过本次遍历,但剩余元素仍然会继续遍历下去。

    • break 只能中断 for 和 while 循环,forEach 函数中使用会报错

    const arr = [1, 2, 3]
    arr.forEach((e, i, a) => {
    console.log(e)
    })

    3.4 for ... in

    `for ... in` 是用来遍历对象(plain object)的,也可以用来遍历数组,但不建议。

    const arr = [1, 2, 3]
    for (let i in arr) {
      console.log(arr[i])
    }

    3.5 map、filter、reduce

    这三类都是为了对数组进行一个操作,然后得到目标结果的数组方法,从功能和语义上来讲,和 forEach 有区别的,不建议混用,尤其是 map 和 forEach。但是也能对数组进行遍历,详细可参考MDN。

    以上就是JS数组遍历的细节以及实现记录的简略介绍,当然详细使用上面的不同还得要大家自己使用过才领会。如果想了解更多,欢迎关注编程之家行业资讯频道哦!

    版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

    相关推荐


    在PHP中进行字符串拼接时,应注意以下几点: 使用 .“运算符进行字符串拼接:在PHP中,可以使用”. 运算符来连接两个字符串。 使用双引号或单引号来包裹字符...
    在Python中,全局变量可以在程序的任何地方进行定义,通常在函数外部进行定义。全局变量可以在整个程序中访问,而不仅仅是在函数内部。要定义一个全局变量,只
    今天小编给大家分享一下电脑显示器上auto指的是什么意思的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考
    本文小编为大家详细介绍“ai建立剪切蒙版后如何移动里面的图片”,内容详细,步骤清晰,细节处理妥当,希望这篇“ai建立剪切蒙版后如何移动里面的图片”文章能帮...
    这篇文章主要讲解了“windows中格式化d盘的后果是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“wind...
    这篇“otf文件有哪些特点”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章...
    这篇文章主要介绍“wpsystem文件夹有什么作用”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“wpsystem文件夹有什
    这篇文章主要介绍了ps单位指的是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇ps单位指的是什么文章都会有所收获,下面我...
    这篇文章主要介绍“ipv6对网速有没有提升”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“ipv6对网速有没有提升”文...
    本文小编为大家详细介绍“islide是什么及有什么作用”,内容详细,步骤清晰,细节处理妥当,希望这篇“islide是什么及有什么作用”文章能帮助大家解决疑惑,下面...
    本篇内容主要讲解“UAC被禁用有哪些影响”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“UAC被禁用有哪些影响”...
    今天小编给大家分享一下svchost.exe可不可以关掉的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,
    这篇文章主要介绍“win10有没有32位版本”,在日常操作中,相信很多人在win10有没有32位版本问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,
    这篇文章主要介绍了vlookup如何引用别的表格数据的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇vlookup如何引用别的表格数据文...
    本文小编为大家详细介绍“.json文件有什么作用”,内容详细,步骤清晰,细节处理妥当,希望这篇“.json文件有什么作用”文章能帮助大家解决疑惑,下面跟着小编的...
    这篇文章主要介绍了vlookup函数的参数是什么意思的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇vlookup函数的参数是什么意思文...
    本篇内容介绍了“wmiprvse.exe程序有什么作用”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情...
    这篇“Windows wifi的ip地址指的是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅...
    今天小编给大家分享一下video接口指的是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大...
    本篇内容介绍了“路由器wps有哪些优缺点”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧...