几种遍历方法中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.length;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'// 迭代map
let 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);
}// 迭代set
let 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 提出。只遍历可迭代对象的数据。

能力甄别

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

关于跳出循环体

在循环中满足一定条件就跳出循环体,或者跳过不符合条件的数据继续循环其它数据。是经常会遇到的需求。常用的语句是break 与 continue

简单的说一下二者的区别,就当复习好了。

  • break语句是跳出当前循环,并执行当前循环之后的语句;

  • continue语句是终止当前循环,并继续执行下一次循环;

注意forEach 与map 是不支持跳出循环体的,其它三种方法均支持。

原理 :查看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 = 'hello
for (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 = 'hello
for (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个只能是各自发挥长处,谁都别想称霸。

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

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

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

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

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

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

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

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

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

  5. 再写循环队列----c++实现

    再写循环队列 class MyCircularQueue {public:/** Initialize your data structure here. Set the size of the qu ...

  6. 通过代码创建一个文件a.txt 然后写一个方法,控制台循环输入内容,然后通过gbk的编码格式保存到 a.txt中(要求可以追加),直到输入exit结束 再写一个方法,要求用字符缓冲流读取a.tx

    通过代码创建一个文件a.txt 然后写一个方法,控制台循环输入内容,然后通过gbk的编码格式保存到 a.txt中(要求可以追加),直到输入exit结束 再写一个方法,要求用字符缓冲流读取a.txt中的 ...

  7. 求求你别再写上千行的类了,试试这些牛逼的重构技巧吧

    欢迎关注方志朋的博客,回复"666"获面试宝典 来源:juejin.cn/post/6844904038383747086 答应我,别再写上千行的类了好吗 最近在对已有项目进行扩展 ...

  8. 漫画:为什么程序员喜欢使用 0 ≤ i 10 左闭右开形式写 for 循环?

    作者 | 漫话编程 来源 | 漫话编程(ID:mhcoding) 当我们想要写一个循环体,期望执行10次的时候,我们会使用以下方式: for (int i=0; i<10; i++){} 可以看 ...

  9. 科技部通知:先看病,再写论文!!!

    来源:科技部 编辑:学妹 科技部通知 先看病,再写论文!!! 想查看更多学术文章 硕博教师生活.论文写作指导 请扫码关注 视学算法 ☟ 点亮"在看"????

最新文章

  1. 使用php读写mysql数据库并显示到网页上
  2. Ubuntu 安装 Docker
  3. 介绍开源的.net通信框架NetworkComms
  4. 阿里云服务器ECS挑选什么样的网站环境
  5. MySQL关于时间设置的注意事项
  6. 有趣的算法(五):一文读懂二叉搜索树的插入、删除
  7. 说说自动学习,是什么样的境界和体验
  8. mac 命令 vim 快捷键
  9. 异数OS 织梦师-Xnign(四)-- 挑战100倍速Nginx,脚踩F5硬件负载均衡
  10. 固定翼飞机数学建模入门(姿态角篇)
  11. CentOs解压缩命令
  12. (中英)作文 —— 标题与小标题
  13. 【工具】Alfred2.5.1使用全攻略!神一样的Mac app!(神一样的搜索)(推荐)
  14. 笔记本电脑计算机无法显示u盘,小编告诉你为什么笔记本电脑识别不了u盘
  15. 通过仿淘宝静态网页我的总结
  16. HTMLCSS仿京东详情页静态页面制作总结
  17. 学习笔记----层次分析法
  18. gitlab:不能中文搜索
  19. 【技术美术】千人千面如何炼成 技术讲解捏脸系统设计原理
  20. 破解root密码,简单粗暴—干就完了!!!

热门文章

  1. 深度学习(二)theano学习笔记(1)环境搭建
  2. Deep Learning论文笔记之(五)CNN卷积神经网络代码理解
  3. 深度学习目标检测系列:RCNN系列算法图解
  4. Firefox联手Chrome合作开发网页VR标准
  5. 前端之JavaScript 补充
  6. .Net开发笔记(十九) 创建一个可以可视化设计的对象
  7. Android仿微信图片上传,可以选择多张图片,缩放预览,拍照上传等
  8. ASP.NET完整打包卸载更新攻略(By Installshield 2010)【转】
  9. android json解析及简单例子
  10. WebSocket的事件触发机制