How to use Reduce in JavaScript

Published on May 09, 2020

|

views -

0

Profile photo of me

Abdulla Aidhaan

Telegram Enthusiast

The reduce function of an `Array` is one of the most versatile functions in JavaScript. With it you can accomplish a lot of the functions from the Array and Math objects. It's job is to start with an array, and reduce those elements into some other type of data... which sounds vague, but you could use it to convert from an array to an object, an array to another array, an array to a number or an array to a boolean. Understanding how it works opens up a world of possibilities.

A few of the Math/Array functions we will cover in this article:

Let's Recreate Reduce

Before we see all of reduce's flexibility, let's understand how it works! Reduce's job is to iterate over each element of an array, calling a function (callback), passing both the current element, and what is called an "accumulator" value. The callback's job is to take the accumulator and the array element, and return the new version of that accmulator.

This may mean taking an accumulator that is a number and returning its value + 1, or taking an accumulator which is an array and returning that array with an additional element on the end. It's up to you, depending what type of data you would like to reduce your array to.

We also need an initial value, which will be the first value our accumulator takes before the callback is ever called.

Javascript

function reduce(array, callback, initial) {
  // start our accumulator off as the initial value
  let acc = initial
  // iterate over each element in the array
  for (let i = 0; i < array.length; i++) {
    // pass the accumulator and current element to callback function
    // override the accumulator with the callback's response
    acc = callback(acc, array[i], i)
  }
  // return the final accumulator value
  return acc
}

result = reduce([1, 2, 3], (acc, num) => acc + num, 0)

Counting Array Length

The data we are starting with is an array of objects which represent people:

Javascript

const people = [
  { id: "1", name: "Leigh", age: 35 },
  { id: "2", name: "Jenny", age: 30 },
  { id: "3", name: "Heather", age: 28 },
]

Although `people.length` would be the more performant and better solution, we can count an array's length by using reduce. Our initial value is 0, and each iteration will add 1 to the previous accumulated value.

Javascript

js result = people.reduce((acc, person) => acc + 1, 0) 

Sum Numbers

We can sum numbers using reduce. Much like the length example above, we can start with 0, and instead of adding 1 upon each iteration, we can add the `person.age` property to our accumulated value.

Javascript

result = people.reduce((acc, person) => acc + person.age, 0) 

Mapping with Reduce

Yup... you can map using reduce! In this case our initial value is an empty array, and upon each iteration we can return an array with its previous value, plus the newest value added to the end of our array.

Javascript

result = people.reduce((acc, person) => [...acc, person.name], [])

Array to Object

This is a technique I use all the time when I have an array of some object with an `id`, and I want to easily access these objects by their ID, rather than having to find them in an array each time. By having an object with each person's ID as the key, I can access them using the ID's value.

Javascript

result = people.reduce((acc, person) => {
  return { ...acc, [person.id]: person }
}, {})

Find Max Value

The `Math.max()` function can be immitated using reduce by checking if the current element's value is greater than the accumulator. If it is, that becomes (by returning the value) the new max value, otherwise the previous accumulator (current max value) is returned.

Javascript

result = people.reduce((acc, person) => {
  if (acc === null || person.age > acc) return person.age
  return acc
}, null)

Find Min Value

The `Math.min()` function can be immitated using reduce by checking if the current element's value is less than the accumulator. If it is, that becomes (by returning the value) the new min value, otherwise the previous accumulator (current min value) is returned.

Javascript

result = people.reduce((acc, person) => {
  if (acc === null || person.age < acc) return person.age
  return acc
}, null)

Find Matching Element

The reduce callback function when immitating `Array.prototype.find()` contains three possibilities:

  • The accumulator is not null, meaning we have already found the value, so let's return it.
  • The current array element meets our criteria, so let's return it.
  • By returning null we tell the next iteration that the value has not yet been found.

Javascript

result = people.reduce((acc, person) => {
  if (acc !== null) return acc
  if (person.name === "Leigh") return person
  return null
}, null)

Check If Every Value Matches

When checking if every value matches a specific criteria, we will start with the assumption that every value will match... very optimistic!! If the accumulator becomes false, our callback will continue to return false, since every value must match, and otherwise will return `true` or `false` for the current element.

Javascript

result = people.reduce((acc, person) => {
  if (!acc) return false
  return person.age > 18
}, true)

Check If Some Value Matches

Checking if some value matches a condition in our array involves a bit of pessimism. We'll start off with the assumption of `false`, and our callback function will return true as soon as the accumulator is true for the first time, and otherwise will return `true` or `false` on the current element.

Javascript

result = people.reduce((acc, person) => {
  if (acc) return true
  return person.age > 18
}, false)

Group and Count Occurrences

As we've seen in the "Array to Object" example above, we can convert an array to an object, but this time we are looking to count the occurrences for a given key, in this case the `status`. Our return value will take the existing accumulated object, adding 1 for the current key's value (or initializing it to 0 if this is the first occurrence).

Javascript

const orders = [
  { id: "1", status: "pending" },
  { id: "2", status: "pending" },
  { id: "3", status: "cancelled" },
  { id: "4", status: "shipped" },
]

result = orders.reduce((acc, order) => {
return { ...acc, [order.status]: (acc[order.status] || 0) + 1 }
}, {})

Flatten Nested Arrays

In this last example we will start with an array of nested arrays, and reduce it into a flattened array of values. Because we will need to recursively reduce (flatten) arrays, we'll need to give our callback function a name (so it can be called within itself).

If the current element is an `Array`, it means we must reduce it until we arrive at at an element that can be appended to the end of the array we are producing. The initial value of our inner reduce call is the current accumulator value.

Javascript

const folders = [
  "index.js",
  ["flatten.js", "map.js"],
  ["any.js", ["all.js", "count.js"]],
]

function flatten(acc, element) {
  if (Array.isArray(element)) {
    return element.reduce(flatten, acc)
  }
  return [...acc, element]
}

result = folders.reduce(flatten, [])