如何理解reduce()?(翻译)

TypeScript/JavaScript函数式编程

翻译自: 《How to understand reduce()?》 - Andrew Zheng

reduce (made by https://carbon.now.sh/)

reduce是另一个函数式编程(Function Programming)概念,它在JavaScript中是作为一个Array方法:Array.prototype.reduce()

reduce() 方法对数组中的每个元素执行一个由您提供的reducer函数(升序执行),将其结果汇总为单个返回值。[1]

上面关于该方法的MDN说明太简单了,有时会引起人们的困惑。实际上,reduce确实非常强大,我认为有两个用例最契合使用它:折叠列表(list)和转换初始值(initial value)

1. reduce参数

首先,让我们看一下reduce中涉及的3个参数:

  • list
  • reducer(一个reduce函数)
  • initial value(有时可以忽略,如果不存在,默认为列表中的第一项)

例如:

function sum(sum: number, item: number): number {
  return sum + item;
}

const list = [1,2,3];

const result1 = Array.prototype.reduce.call(list, sum); // -> 6
const result2 = Array.prototype.reduce.call(list, sum, 1); // -> 7

list是一个数组,reducersum函数以及initial value在第二种情况下初始值为1.

2. 输入/输出形状

Shape of Input/Output

为了更好地理解reduce,让我们来定义一个“ 输入 / 输出 ”形状。我们都知道一个函数接受(多个)输入并生成输出。在纯函数中是没有副作用的,因此可以说,当我们调用纯函数时,它只是一个将输入转换输出的过程。

例如,以下函数输入2个数字并输出1个数字。这看起来像:[], number -> []

function sum(a: number[], b: number): number[] {
  return [...a, b]
}

你可以将一块岩石堆放在一堆岩石上来进行可视化处理:


https://unsplash.com/photos/70zb7HHhspc

现在,考虑到该形状概念,我们可以将reduce的用法分为两种情况:

  1. 折叠列表(list),这看起来像:Object[] -> Object
  2. 转换初始值(initial value),这看起来像:Object -> Object

2.1. 折叠列表(list):Object[] -> Object

这是为了从列表中转换输入并输出单个值。
reduce有时也被成为fold(折叠),我们可以通过下面图片中长毯子折叠成垫子上获得提示:

叠毯子示意图

reduce的大多数示例都属于此类。

案例1:计算数组之和

function sum(sum: number, item: number): number {
  return sum + item;
}

const list = [1,2,3];

const result = Array.prototype.reduce.call(list, sum); // -> 6;

实现过程:

  • 列表(list)中的第一项作为初始值(initial value)
  • 遍历列表(list)
  • 逐一调用reducer函数,并将结果链接到下一个迭代

案例2:数组中找到最大值

基于输入和输出形状,它将一个列表转换为一个值,所以我们还可以这样使用reduce:

function max(a: number, b: number): number {
  return a > b ? a : b;
}

const list = [1,5,3];

const result = Array.prototype.reduce.call(list, max); // -> 5

2.2. 转换初始值(initial value):Object -> Object

除了变换列表(list),我们还可以变换初始值(initial value)。这种情况就像传入然后递归调用reducer函数逐一转换初始值(initial value)并链接输出。

https://twitter.com/danicapatrick/status/508602513789304832

想象一下,你和你的粉丝击掌,你是初始值,粉丝们是列表,击掌动作是reducer函数

案例3:用替换选项列表替换uri参数

假如你有一个uri https://abc.com/:id/docs/:docId/history并且你想将:id替换成用户id。

考虑一下上面提到的形状函数是将对象(uri)转换为另一个uri(替换参数),因此我们可以这样做:

const replacementOptions = [
  {replace: ":id", to: "SJDUEB"},
  {replace: ":docId", to: "12345"}
]

const uri = "https://abc.com/ :id /docs/ :docId /history";

function reduce(uri: string, option: any):string {
  return uri.replace(option.replace, option.to);
}

const reult = Array.prototype.reduce.call(
  replacementOptions, // list
  reduce, // reducer
  uri // initial value
)

在这种情况下,我们可以看到reducer函数根据列表(replacementOptions)转换了初始值(uri)

⚠️请注意,它并没有改变原列表数组。

案例4:编写一个简单的Redux实现

Redux是一个状态管理库,它具有全局状态对象,并且每一次操作状态都会被执行一次reducer函数

const initialState = { sum:0, history:[] }

function reduce(state, action){
  return {
    sum: state.sum + action.n,
    history: [...state.history, action.name]
  }
}

const action1 = {
  name: 'action1',
  n: 1
};
const action2 = {
  name: 'action2',
  n: 2
};
const list = [action1, action2];

const finalState = Array.prototype.reduce.call(
  list,
  reduce,
  initialState
); 

console.log(finalState); {sum:3, history:['action1', 'action2']}

再一次,你可以看到我们只是转换了初始值(initial value),而不是列表list。

案例5: 将多个数据变成一个数据(参数完整使用到的案例)

  • callback执行数组中每个值(如果没有提供initialValue则第一个值除外)的函数,包含四个参数:

    • accumulator 累加器累计回调的返回值
      它是上一次调用回调时返回的累计值(或initialValue

    • currentValue 数组中正在处理的元素

    • currentIndex 数组中正在处理的当前元素的索引
      可选。如果提供了initialValue,则起始索引号为0,否则从索引1起始

    • array调用reduce()的数组可选

  • initialValue 作为第一次调用callback是的第一个参数的值。
    可选。如果没有提供初始值,则将使用数组中的第一个元素。在没有初始值的空数组上调用reduce将报错

//输出为 {name:'Niko', age:25}
const keys = ['name', 'age'];
const values = ['Niko', 25];

keys.reduce(function(accumulator, currentValue, currentIndex){
  accumulator[currentValue] = values[currentIndex];
  return accumulator;
},{})

案例6: redux中的compose函数

compose 从右到左来组合多个函数

function compose(...funcs){
    if(funcs.length === 0){
        return arg => arg
    }

    if(funcs.length === 1){
         return funcs[0]
    }

    return funcs.reduce((a,b)=> (...args) => a(b(...args)))
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容