reduce lodash.reduce实现原理是什么

其他教程   发布日期:2023年07月13日   浏览次数:391

本篇内容主要讲解“reduce lodash.reduce实现原理是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“reduce lodash.reduce实现原理是什么”吧!

基本实现

实现思路:

  • 判断是否有初始值,因为有初始值和没有初始值对回调函数(reducer)执行的次数是有影响的。

  • 遍历数组

  • 组合参数传递给 reducer 进行执行

  • 获取到第三步返回值的时候,要把返回值存储起来,在下一次便利的时候作为reducer第一个参数来替换初始值。

  • 返回最终计算的value值

  1. function reduce(array, reducer, initialValue = null) {
  2. let value = initialValue === null ? array[0] : initialValue; // 思路1
  3. let startIndex = initialValue === null ? 1 : 0; // 思路1
  4. for(let i = startIndex; i < array.length; i++) { // 思路 2
  5. const item = array[i]
  6. const res = reducer(value, item, i) // 思路3
  7. value = res; // 思路4
  8. }
  9. return value; // 思路5
  10. }

测试一下:

  1. console.log(reduce([1,2,3], (a, b) => (a + b), 0)) // 6
  2. console.log(reduce([1,2,3], (a, b) => (a + b))) // 6

看起来是不是挺简单的,代码其实还可以更简洁一点:

  1. function reduce(array, reducer, value = null) {
  2. value = value === null ? array[0] : value;
  3. for(let i = null ? 1 : 0; i < array.length; i++) {
  4. value = reducer(value, array[i], i);
  5. }
  6. return value;
  7. }

lodash 中的 reduce 实现有何不同?

lodash中 的 reduce 不仅可以对数组生效,也可以对普通 object 、类数组对象生效。

不过也针对数组其实单独实现了一个

  1. arrayReduce
函数,不过没有对外。

来看一下

  1. reduce
  1. arrayReduce
源码
  1. function reduce(collection, iteratee, accumulator) {
  2. const func = Array.isArray(collection) ? arrayReduce : baseReduce
  3. const initAccum = arguments.length < 3
  4. return func(collection, iteratee, accumulator, initAccum, baseEach)
  5. }
  6. function arrayReduce(array, iteratee, accumulator, initAccum) {
  7. let index = -1
  8. const length = array == null ? 0 : array.length
  9. if (initAccum && length) {
  10. accumulator = array[++index]
  11. }
  12. while (++index < length) {
  13. accumulator = iteratee(accumulator, array[index], index, array)
  14. }
  15. return accumulator
  16. }

看得懂吗?不理解的话看下面一份代码,我把非数组类型的代码去掉,再调一下变量命名和新增注释:

  1. function reduce(array, reducer, value) {
  2. const noInitialValue = arguments.length < 3 // 用参数的数量来判断是否有初始值
  3. let index = -1 // 遍历索引 - 1,因为下面 while 循环前先加了 1
  4. const length = array == null ? 0 : array.length // 判断数组是否存在和缓存数组长度
  5. // 这个if 语句中做了我上面思路1中初始值的问题和遍历次数的问题
  6. if (noInitialValue && length) { // && length 判断了数组是否为空
  7. value = array[++index] // 没有有初始值,则取数组中第一为,注意 index 变成了0,下面 while 循环前会先加 1,因此循环次数会少一次。
  8. }
  9. while (++index < length) {
  10. value = reducer(value, array[index], index, array)
  11. }
  12. return value
  13. }

可以看出其实大部分逻辑还是和前面的简单实现差不多,不过考虑更全一些,有值得借鉴的地方:

  • 参数判断逻辑更有力,不管外界传递了第三个参数是啥,都说明有初始值

  • 考虑了数组不存在或者为空的情况

下面我们再看一下,去除数组相关的代码来看看针对其他对象类型怎么处理的。

  1. function reduce(collection, iteratee, accumulator) {
  2. const func = baseReduce;
  3. const initAccum = arguments.length < 3
  4. return func(collection, iteratee, accumulator, initAccum, baseEach)
  5. }

其他类型的都会教给

  1. baseReduce
函数去处理。
  1. // baseReduce
  2. function baseReduce(collection, iteratee, accumulator, initAccum, eachFunc) {
  3. // 使用外部传递进来的遍历方法进行遍历对象,然后传递了一个 callback 给 eachFunc
  4. eachFunc(collection, (value, index, collection) => {
  5. // 初始值设置,
  6. accumulator = initAccum
  7. ? (initAccum = false, value)
  8. : iteratee(accumulator, value, index, collection)
  9. })
  10. return accumulator
  11. }

使用外部传递进来的遍历方法进行遍历对象,然后传递了一个 callback 给 eachFunc,来执行 iteratee (也就是前面说的reducer),callback 内部的代码就和前面 for 循环或者 while 循环的代码类似的,就是组合参数传递给 reducer 进行执行,不过直接看可能有点不好理解中,了解了原理再来看应该可以理解,注意事项:

  • initAccum 为 false 时,说明有初始值,直接执行 iteratee。

  • initAccum 为 true,说明没有初始值,需要添加初始值,因此第一次循环就是赋值给初始值,然后把 initAccum 设置为false,没有进行执行 iteratee,比没有初始值少一次执行,符合逻辑。

  1. eachFunc
用的是
  1. reduce
中传递进来的
  1. baseEach
,内部主要就是对对象属性进行遍历的操作,然后把属性值和索引以及对象本身传递给 callback,稍微需要注意的就是可能遇到类数组的对象,为了保证顺序,使用类数组放入索引进行遍历,而其他对象并不能保证属性的传递顺序,可以再看一下baseEach实现的代码:
  1. function baseEach(collection, iteratee) {
  2. if (collection == null) {
  3. return collection
  4. }
  5. // 不是类数组则使用 baseForOwn 处理
  6. if (!isArrayLike(collection)) {
  7. return baseForOwn(collection, iteratee)
  8. }
  9. const length = collection.length
  10. const iterable = Object(collection) // 使用arguments测试了一下,好像没啥作用
  11. let index = -1
  12. // 遍历类数组
  13. while (++index < length) {
  14. if (iteratee(iterable[index], index, iterable) === false) {
  15. break
  16. }
  17. }
  18. return collection
  19. }

不是

  1. isArrayLike
的对象遍历与本篇文章的内容没有啥关系了,因此就不深入了。

以上就是reduce lodash.reduce实现原理是什么的详细内容,更多关于reduce lodash.reduce实现原理是什么的资料请关注九品源码其它相关文章!