JS - 基础学习(6): reduce() 方法    

在前后端数据交互的过程中,为了能够减少ajax请求次数,减轻带宽压力,后端往往会将当前接口所需的参数以对象的形式集体返回。这样就导致一个问题:对象内字段属性过多,而有些小组件功能又压根不需要这么一个大对象参数(主要是对象属性过多,难得理,也懒得找),这时就需要对这个大对象做再加工处理。

比如:将小组件所需的字段属性拎出来单独再封装成一个小对象。一般情况下我的写法是:

let retData = {a: 1, b: 2, c: 3, d: 4, e: 5, f: 6, g: 7, h: 8, j: 9, k: 10, m: 11, n: 12};
let tempData = {
    a: retData.a,
    c: retData.c,
    d: retData.d,
    g: retData.g,
};

但是这种写法却很麻烦,要一个属性一个属性的添加,于是就突发奇想地想优化一下这种写法。苦思冥想该怎么玩,即要写起来简单,又要显得 big 高,所以就用到了reduce()方法。

function handlingObjectProperty(propertyList, obj) {    return propertyList.reduce((iter, val) => {        if (val in obj) {
            iter[val] = obj[val];
        }        return iter;
    }, {});
}

let retData = {a: 1, b: 2, c: 3, d: 4, e: 5, f: 6, g: 7, h: 8, j: 9, k: 10, m: 11, n: 12};
let tempData = handlingObjectProperty(['a', 'c', 'f', 'k'], retData);

JS - 基础学习(6): reduce() 方法_reduce,同样得将所需的字段属性拎出来封装成一个小对象,这样就不需要一个属性一个属性地添加了,方便了很多,一行代码就搞定。

 

reduce方法的定义和用法

reduce() 方法接收一个回调函数作为处理器,数组中的每一个元素(从左到右)【不包括数组中被删除或从未被赋值的元素】依次执行该回调函数,并最终计算为一个值或一个对象。reduce() 可以作为一个高阶函数,用于函数的 compose。

reduce() 方法回调函数接受四个参数:提供的初始值(或上一次调用回调函数返回的值),当前被处理的元素,当前正被处理元素在数组中的索引,调用 reduce方法 的数组。

语法:

array.reduce(function(total, currentValue, currentIndex, arr), initialValue)// 或array.reduce(callback,[initialValue])

参数:

JS - 基础学习(6): reduce() 方法_reduce_02

initialValue参数解析

  1、不传 initialValue参数

let testArr = [1, 2, 3, 4];
let sum = testArr.reduce((prev, currentValue, currentIndex, testArr) => {
    console.log(prev, currentValue, currentIndex);    return prev + currentValue;
});
console.log(`sum: ${sum}`);

JS - 基础学习(6): reduce() 方法_reduce_03 输出结果。

  从控制台打印结果可以看出:由于没有给 reduce方法传入 initialValue参数,所以 reduce方法的回调函数 total参数 以 testArr数组的第一个元素作为初始值,currentValue则从第二个元素开始,因而 currentIndex从1开始。所以,虽然 testArr数组的长度是4,但是 reduce方法只循环了 3次。

  2、传入 initialValue参数

let testArr = [1, 2, 3, 4];
let sum = testArr.reduce((prev, currentValue, currentIndex, testArr) => {
    console.log(prev, currentValue, currentIndex);    return prev + currentValue;
}, 0);
console.log(`sum: ${sum}`);

JS - 基础学习(6): reduce() 方法_reduce_04输出结果。

  这次给 reduce方法 传入了 initialValue参数,所以 total参数以 initialValue参数 作为初始值,currentValue则自然而然从第一个元素开始,currentIndex也就从0开始。所以,testArr数组的长度是4,reduce方法也就循环了 4次。

  3、如果array数组是个空数组,且未传入 initialValue参数。此时代码执行会报错,回调函数也不会被执行。

let testArr = [];
let sum = testArr.reduce((prev, currentValue, currentIndex, testArr) => {
    console.log(prev, currentValue, currentIndex);    return prev + currentValue;
});// index_reduce.js:18 Uncaught TypeError: Reduce of empty array with no initial value

JS - 基础学习(6): reduce() 方法_JS_05

  4、如果array数组是个空数组,但传入 initialValue参数。此时代码能顺利执行,回调函数也会执行,只是由于是空数组,没得结果而已。

let testArr = [];
let sum = testArr.reduce((prev, currentValue, currentIndex, testArr) => {
    console.log(prev, currentValue, currentIndex);    return prev + currentValue;
}, 0);

  所以:不管什么情况下,传入 initialValue参数值,都会让代码执行更安全,也更好维护。

基本用法:

  reduce()方法的基本用法就是把回调函数做计算器使用,对数组或数列进行求和、求积。

let testArr = [1, 2, 3, 4];
let sum = testArr.reduce((prev, cur) => {    return prev + cur
}, 0);
console.log(`sum: ${sum}`);                            // 10let pro = testArr.reduce((prev, cur) => {    return prev * cur
}, 1);                                                 // 求积时,initialValue不能初始化为0console.log(`pro: ${pro}`);                          // 24

其他用法:

  1、计数数组中每个元素、字符串中每个字符出现的总次数

// 获取数组内、字符串内某个元素、字符出现总次数function getFrequency(parList) {    return parList.reduce((pre, cur) => {        if (cur in pre) {
            pre[cur]++
        } else {
            pre[cur] = 1
        }        return pre
    }, {});
}

let iterators = ['reduce', 'map', 'for', 'forOf', 'forEach', 'reduce', 'for'];
let testStr = 'reducemapforforOfforEachreducefor';
console.log(getFrequency(iterators));                      // {reduce: 2, map: 1, for: 2, forOf: 1, forEach: 1}console.log(getFrequency(testStr.split('')));              // {r: 6, e: 4, d: 2, u: 2, c: 3, …}    注:字符串不能直接被reduce,所以先把字符串分割成字符数组

  2、数组,字符串字符去重

// 数组去重、字符串内字符去重function deduplication(parList) {    return parList.reduce((pre, cur) => {        if(!pre.includes(cur)){            return pre.concat(cur)
        }else{            return pre
        }
    }, []);
}

let iterators = ['reduce', 'map', 'for', 'forOf', 'forEach', 'reduce', 'for'];
let testStr = 'reducemapforforOfforEachreducefor';
console.log(deduplication(iterators));                     // ["reduce", "map", "for", "forOf", "forEach"]console.log(deduplication(testStr.split('')).join(''));    // reducmapfoOEh    注:字符串不能直接被reduce,故先将字符串分割成字符数组,reduce完成后,再格式化成字符串

  3、数组降维(又名:数组的扁平化)

// 下面 reduce回调函数内对 pre参数的操作,只能用concat方法,不能用push方法。因为,push方法虽然也将 cur添加到了 pre末尾,但是返回的却是 pre当前的长度,而不是当前 pre的值。function dimensionalityReduction(parList) {    return parList.reduce((pre, cur) => {        return pre.concat(Array.isArray(cur) ? dimensionalityReduction(cur) : cur)
    }, []);
}

let testArr = [[1, 2], [3, [4]], [[5], 6]];
console.log(dimensionalityReduction(testArr));             // [1, 2, 3, 4, 5, 6]

  4、根据需求对 对象属性的操作

  41:如本文开头所述的,从一个大对象内将所需的字段属性拎出来单独再封装成一个小对象;42:对 对象数组(后端返回的表格数据,一般都是对象数组) 内各个对象特定属性值进行特定处理:如根据状态值不同显示不同提示语、数据格式转换、数据逻辑处理等。

// 42、数据逻辑处理function processData(parList, attr) {    return parList.reduce((pre, cur) => {        return pre + cur[attr];
    }, 0);
}

let testArr = [
    {month: 1, sales: 85},
    {month: 2, sales: 30},
    {month: 3, sales: 40},
];
console.log(`2020年第一季度销售总额: ${processData(testArr, 'sales')}万元`);        // 2020年第一季度销售总额: 155万元

  对于对象数组而言,reduce方法适用于 将每个对象元素内指定字段属性单独拿出来做处理,然后返回一个值或一个对象,如本文开头的字段属性重新封装,这里的数据累加等。而不是对这个数组本身的处理,对于这些对象数组本身的处理,for,forEach,map等方法更实用。

最后

  reduce方法可以实现的东西,很多时候for循环,forEach方法、甚至map方法都可以实现,那为啥要用reduce呢。个人觉得吧,这个无关啥逼格什么的,仅仅只是想让代码更简洁,功能逻辑简单化,同时也让代码多元化,不让之前学过的东西束之高阁,喂灰尘罢了。