几种遍历方法中for执行最快,它没有任何额外的函数调用栈和上下文。但在实际开发中我们要结合语义话、可读性和程序性能,去选择究竟使用哪种方案。下面来看for, foreach, map, for...in, for...of五种方法现场battle。自我介绍for我是最早出现的一方遍历语句,在座的各位需称我一声爷爷。我能满足开发人员的绝大多数的需求。

// 遍历数组let arr = [1,2,3];for(let i = 0;i < arr.length;i++){console.log(i) // 索引,数组下标console.log(arr[i]) // 数组下标所对应的元素}// 遍历对象let profile = {name:"April",nickname:"二十七刻",country:"China"};for(let i = 0, keys=Object.keys(profile); i < keys.length;i++){console.log(keys[i]) // 对象的键值console.log(profile[keys[i]]) // 对象的键对应的值}// 遍历字符串let str = "abcdef";for(let i = 0;i < str.length ;i++){console.log(i) // 索引 字符串的下标console.log(str[i]) // 字符串下标所对应的元素}// 遍历DOM 节点let articleParagraphs = document.querySelectorAll('.article > p');for(let i = 0;i    articleParagraphs[i].classList.add("paragraph");// 给class名为“article”节点下的 p 标签添加一个名为“paragraph” class属性。}

forEach

我是ES5版本发布的。按升序为数组中含有效值的每一项执行一次 callback 函数,那些已删除或者未初始化的项将被跳过(例如在稀疏数组上)。我是 for 循环的加强版。

// 遍历数组let arr = [1,2,3];arr.forEach(i => console.log(i))// logs 1// logs 2// logs 3// 直接输出了数组的元素//遍历对象let profile = {name:"April",nickname:"二十七刻",country:"China"};let keys = Object.keys(profile);keys.forEach(i => {console.log(i) // 对象的键值console.log(profile[i]) // 对象的键对应的值})

map

我也是ES5版本发布的,我可以创建一个新数组,新数组的结果是原数组中的每个元素都调用一次提供的函数后的返回值。

let arr = [1,2,3,4,5];let res = arr.map(i => i * i);console.log(res) // logs [1, 4, 9, 16, 25]

for...in枚举

我是ES5版本发布的。以任意顺序遍历一个对象的除Symbol以外的可枚举属性。

// 遍历对象let profile = {name:"April",nickname:"二十七刻",country:"China"};for(let i in profile){let item = profile[i];console.log(item) // 对象的键值console.log(i) // 对象的键对应的值// 遍历数组let arr = ['a','b','c'];for(let i in arr){let item = arr[i];console.log(item) // 数组下标所对应的元素console.log(i) // 索引,数组下标// 遍历字符串let str = "abcd"for(let i in str){let item = str[i];console.log(item) // 字符串下标所对应的元素console.log(i) // 索引 字符串的下标}

for...of迭代

我是ES6版本发布的。在可迭代对象(包括 Array,Map,Set,String,TypedArray,arguments 对象等等)上创建一个迭代循环,调用自定义迭代钩子,并为每个不同属性的值执行语句。

// 迭代数组数组let arr = ['a','b','c'];for(let item of arr){console.log(item)}// logs 'a'// logs 'b'// logs 'c'// 迭代字符串let str = "abc";for (let value of str) {console.log(value);}// logs 'a'// logs 'b'// logs 'c'// 迭代maplet iterable = new Map([["a", 1], ["b", 2], ["c", 3]]for (let entry of iterable) {console.log(entry);}// logs ["a", 1]// logs ["b", 2]// logs ["c", 3]// 迭代map获取键值for (let [key, value] of iterable) {console.log(key)console.log(value);}// 迭代setlet iterable = new Set([1, 1, 2, 2, 3, 3,4]);for (let value of iterable) {console.log(value);}// logs 1// logs 2// logs 3// logs 4// 迭代 DOM 节点let articleParagraphs = document.querySelectorAll('.article > p');for (let paragraph of articleParagraphs) {    paragraph.classList.add("paragraph");// 给class名为“article”节点下的 p 标签添加一个名为“paragraph” class属性。}// 迭代arguments类数组对象(function() {for (let argument of arguments) {console.log(argument);  }})(1, 2, 3);// logs:// 1// 2// 3// 迭代类型数组let typeArr = new Uint8Array([0x00, 0xff]);for (let value of typeArr) {console.log(value);}// logs:// 0// 255

经过第一轮的自我介绍和技能展示后,我们了解到:

  • for语句是最原始的循环语句。定义一个变量i(数字类型,表示数组的下标),按照一定的条件,对i进行循环累加。条件通常为循环对象的长度,当超过长度就停止循环。因为对象无法判断长度,所以搭配Object.keys()使用。
  • forEach ES5 提出。自称是for语句的加强版,可以发现它比for语句在写法上简单了很多。但是本质上也是数组的循环。forEach每个数组元素执行一次 callback 函数。也就是调用它的数组,因此,不会改变原数组。返回值是undefine
  • map  ES5 提出。给原数组中的每个元素都按顺序调用一次  callback 函数。生成一个新数组,不修改调用它的原数组本身。返回值是新的数组。
  • for...in  ES5 提出。遍历对象上的可枚举属性,包括原型对象上的属性,且按任意顺序进行遍历,也就是顺序不固定。遍历数组时把数组的下标当作键值,此时的i是个字符串型的。它是为遍历对象属性而构建的,不建议与数组一起使用。
  • for...of ES6 提出。只遍历可迭代对象的数据。

能力甄别

作为一个程序员,仅仅认识他们是远远不够的,在实际开发中鉴别他们各自的优缺点。因地制宜的使用他们,扬长避短。从而提高程序的整体性能才是能力之所在。

关于跳出循环体

在循环中满足一定条件就跳出循环体,或者跳过不符合条件的数据继续循环其它数据。是经常会遇到的需求。常用的语句是breakcontinue。简单的说一下二者的区别,就当复习好了。

  • break语句是跳出当前循环,并执行当前循环之后的语句;
  • continue语句是终止当前循环,并继续执行下一次循环;

注意forEachmap是不支持跳出循环体的,其它三种方法均支持。原理 :查看forEach实现原理,就会理解这个问题。

Array.prototype.forEach(callbackfn [,thisArg]{}

传入的function是这里的回调函数。在回调函数里面使用break肯定是非法的,因为break只能用于跳出循环,回调函数不是循环体。在回调函数中使用return,只是将结果返回到上级函数,也就是这个for循环中,并没有结束for循环,所以return也是无效的。map()同理。

map()链式调用

map()方法是可以链式调用的,这意味着它可以方便的结合其它方法一起使用。例如:reduce(), sort(), filter()等。但是其它方法并不能做到这一点。forEach()的返回值是undefined,所以无法链式调用。

// 将元素乘以本身,再进行求和。let arr = [1, 2, 3, 4, 5];let res1 = arr.map(item => item * item).reduce((total, value) => total + value);console.log(res1) // logs 55 undefined"

for...in会遍历出原型对象上的属性

Object.prototype.objCustom = function() {};Array.prototype.arrCustom = function() {};var arr = ['a', 'b', 'c'];arr.foo = 'hellofor (var i in arr) {    console.log(i);}// logs// 0// 1// 2// foo// arrCustom// objCustom

然而在实际的开发中,我们并不需要原型对象上的属性。这种情况下我们可以使用hasOwnProperty()方法,它会返回一个布尔值,指示对象自身属性中是否具有指定的属性(也就是,是否有指定的键)。如下:

Object.prototype.objCustom = function() {};Array.prototype.arrCustom = function() {};var arr = ['a', 'b', 'c'];arr.foo = 'hellofor (var i in arr) {    if (arr.hasOwnProperty(i)) {        console.log(i);    }}// logs// 0// 1// 2// foo// 可见数组本身的属性还是无法摆脱。此时建议使用 forEach

对于纯对象的遍历,选择for..in枚举更方便;对于数组遍历,如果不需要知道索引for..of迭代更合适,因为还可以中断;如果需要知道索引,则forEach()更合适;对于其他字符串,类数组,类型数组的迭代,for..of更占上风更胜一筹。但是注意低版本浏览器的是配性。

性能

有兴趣的读者可以找一组数据自行测试,文章就直接给出结果了,并做相应的解释。

for > for-of > forEach > map > for-in
  • for 循环当然是最简单的,因为它没有任何额外的函数调用栈和上下文;
  • for...of只要具有Iterator接口的数据结构,都可以使用它迭代成员。它直接读取的是键值。
  • forEach,因为它其实比我们想象得要复杂一些,它实际上是array.forEach(function(currentValue, index, arr), thisValue)它不是普通的 for 循环的语法糖,还有诸多参数和上下文需要在执行的时候考虑进来,这里可能拖慢性能;
  • map() 最慢,因为它的返回值是一个等长的全新的数组,数组创建和赋值产生的性能开销很大。
  • for...in需要穷举对象的所有属性,包括自定义的添加的属性也能遍历到。且for...in的key是String类型,有转换过程,开销比较大。

总结

在实际开发中我们要结合语义话、可读性和程序性能,去选择究竟使用哪种方案?

  • 如果你需要将数组按照某种规则映射为另一个数组,就应该用 map。

  • 如果你需要进行简单的遍历,用 forEach 或者 for of。

  • 如果你需要对迭代器进行遍历,用 for of。

  • 如果你需要过滤出符合条件的项,用 filterr。

  • 如果你需要先按照规则映射为新数组,再根据条件过滤,那就用一个 map 加一个 filter。

总之,因地制宜,因时而变。千万不要因为过分追求性能,而忽略了语义和可读性。在您的统治之下,他们5个只能是各自发挥长处,谁都别想称霸。

来源:juejin.im/post/5ea63f3ef265da47b177b4b6

 往期推荐 

for循环如果先--_如果再写for循环,我就锤自己!相关推荐

  1. python遍历循环怎么理解_聊聊python中的循环遍历

    python之循环遍历 关于循环遍历大家都知道,不外乎for和while,今天我在这写点不一样的循环和遍历.在实践中有时会遇到删除列表中的元素,那么循环遍历列表删除指定元素该怎么做呢? 还是直接上代码 ...

  2. 以后要是再写for循环,我就捶自己

    点击上方 好好学java ,选择 星标 公众号 重磅资讯.干货,第一时间送达 今日推荐:干掉 Navicat:这个 IDEA 的兄弟真香!个人原创100W+访问量博客:点击前往,查看更多  请 听 题 ...

  3. 如果再写for循环,我就锤自己!

    点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 几种遍历方法中for执行最快,它没有任何额外的函数调用栈和上下文. ...

  4. 如果再写 for 循环,我就锤自己!

    几种遍历方法中for执行最快,它没有任何额外的函数调用栈和上下文.但在实际开发中我们要结合语义话.可读性和程序性能,去选择究竟使用哪种方案.下面来看for , foreach , map , for. ...

  5. for循环下标 shell_如果再写for循环,我就锤自己

    几种遍历方法中for执行最快,它没有任何额外的函数调用栈和上下文.但在实际开发中我们要结合语义话.可读性和程序性能,去选择究竟使用哪种方案.下面来看for , foreach , map , for. ...

  6. controller层要写什么_别再写满屏的try-catch了,真丑,全局异常处理不会吗?

    本文讲得比较细,所以篇幅较长.请认真读完,希望读完后能对统一异常处理有一个清晰的认识. 背景 软件开发过程中,不可避免的是需要处理各种异常,就我自己来说,至少有一半以上的时间都是在处理各种异常情况,所 ...

  7. vb 循环放音乐_为何洒水车一直无限循环播放《兰花草》这首歌呢?

    展开全部 在城市中,洒水车2113可谓是一道亮丽的风景.5261洒水车的作用4102实在是太多了,首先可以用来清洁道路,1653然后可以给道路上的花花草草以及高大的树木浇水.到了夏天的时候洒水车还肩负 ...

  8. python循环代码优化技巧_记一次优化python循环代码逻辑的过程

    问题描述: 一个含有30W元素的列表A,列表的元素都是字符串,现在要循环10W次,每次都要判断一下B字符串是否存在这个A列表里面,有什么优化策略吗? 问题现状: 如果用普通的逻辑来写, 程序会类似下面 ...

  9. python跳回循环开始位置_如何回到python中循环的开始?

    我的代码:b="y" ol=[] #operations list OPERATIONS = ["-", "+", "*" ...

最新文章

  1. 一个小小的AI训练营竟然卧虎藏龙
  2. mac设置计算机用户名,如何更改macbook用户名_高手教你更改macbook用户名的方法-系统城...
  3. 10 个免费的 C/C++ 集成开发环境
  4. 蒙特卡罗方法验证凯利公式
  5. py脚本得到Python的版本
  6. 吴恩达 coursera ML 第五课总结+作业答案
  7. php 接口继承,详细对比php中类继承和接口继承
  8. windows 临界区
  9. easy-excel导入导出excel(待完善)
  10. UnityShader中的Queue
  11. 需求与商业模式分析-2-商业模式类型
  12. JVM内存模型和结构
  13. PIC单片机入门教程(一)—— 准备工作
  14. java程序员转正述职报告PPT
  15. C#-Winform - 调用笔记本摄像头实现拍照并保存
  16. JAVA高并发学习笔记(二) 多线程基础
  17. 网园网络电视 v1.2 官方
  18. Spring Security CSRF防御源码分析
  19. 测试版ios15怎么信任软件,ios15的信任文件在哪?ios15信任授权在哪里设置?
  20. reviewer中文_审稿意见回复模板,中文

热门文章

  1. oracle 去重_超详细的四类数据库去重实现方案汇总,值得收藏
  2. Nacos简介和安装
  3. 学习记录——背包问题基础公式解释回顾
  4. vivado 启动过程中报错
  5. python朋友圈动态_如何利用Python网络爬虫爬取微信朋友圈动态--附代码(下)
  6. 我的世界服务器物品解绑定,我的世界更方便控制VIP物品 灵魂绑定插件分享
  7. efm32芯片电压_解读GP21+EFM32低功耗热量表电路
  8. 解决报错ModuleNotFoundError: No module named ‘fastText‘
  9. 解决pytorch CrossEntropyLoss报错RuntimeError: 1D target tensor expected, multi-target not supported
  10. supersu二进制更新安装失败_Q音直播编译优化与二进制集成方案