JavaScript中遍历我已经在《javascript的几种常见遍历数据结构的语法》中讲过了,但是数组有几个自带的遍历方法map、forEach、reduce、filter他们的区别是什么呢?本文简单的总结一下,让你装逼无误。
意义不一样
意义不一样,一样的话闲的蛋疼干嘛搞出这么多方法呢?意义不一样代表着使用场景不一样。
forEach:依次查看元素
map:构建元素的某种映射关系,比如每一个元素在原始值基础上自加1,意味着一定要有一个返回值,返回值是该元素的新值,元素本身会改变,而且整个数组也因此改变
reduce:累加,这个规则比较麻烦,但总结起来就是,它得到的是一个值,而其他的方法得到的是一个数组,那么这个累加起来的值怎么确定呢?不是每个元素相加,而是遍历到最后一次执行回调函数的结果,所以这个方法比较麻烦,下面会讲
filter:过滤,可以用来删掉一些不需要的元素
结果(返回值)不一样
上面的阐述你大概也看到了,forEach()结果是undefined;map和filter结果是Array,但是filter不会改变原始数组,只会返回一个新数组;reduce结果是单个值。
回调函数不一样
回调函数不一样是指回调函数的参数和返回值不一样,这节就详细讲下。
参数不一样
array.forEach(function(value,index,array){},this);
array是array本身,this是forEach的第二个参数,相当于对回调函数执行bind(this)操作,这样回调函数里面的this就可以被替换。
array.reduce(function(prev,value,index,array){},init);
先讲init,遍历一开始的时候,会执行回调函数,这个时候回调函数的prev值不存在,所以需要设定一个init,相当于把init赋值给这个时候的prev。当遍历进入第二个循环的时候,prev就是上一次执行回调函数的返回值。相当于把上一次执行的结果传给第二次执行,类似于一个临时变量,否则怎么实现累加呢?
array.map(function(value,index,array){},this);
array.filter(function(value,index,array){},this);
从参数的角度讲,map、filter和forEach是一样的,不需要额外讲。
返回值不同
前面讲的结果(返回值)是指方法的返回值,这里要讲的是回调函数的返回值。
forEach的回调函数不需要返回值,在jQuery的each方法中使用return false来打断循环,但在forEach里面,return无效,无论如何,都会完全遍历整个数组的所有元素。
reduce的回调函数返回值是一个单纯的值,如果是求数值的和的话,就是一个数字。
map的回调函数返回值也是一个值,但这个值将直接替换当前遍历到的这个元素,从而引发原始数组的改变。如果元素是一个对象的话,应该尽可能做到对象结构的不变,只是属性或方法内容的改变。
filter的回调函数的返回值是一个boolean值,为true时保留这个元素,为false时filter的返回值中不包括这个元素。(filter不会对数组自己进行修改,而是返回一个新数组。)
使用场景不同
forEach用于看,map用于改,filter用于删,reduce用于统计。
细节使用上的不同
forEach除了上面讲到的return无效,continue/break等也无效。
map使用的时候要特别当心,因为在它的内部可以修改数组,这一点和forEach完全不同,你必须在末尾使用return返回一个值。
了通过return改变数组,直接通过其他方式也可以改变数组比如你想改变当前这个元素的id属性
array.map(item => item.id += 2016);
通过这段代码可以改变item.id。
filter就是注意返回值true/false就可以了,同时注意它返回一个新数组,而不会修改元素数组。但是如果你在filter内部执行像上面那段代码一样的操作,其实还是会改变原始数组。
reduce用法差异比较大,细节基本上都不同。