Friday, 23 April, 2021 UTC


Summary

Array reduce is a pre-built function in JavaScript notorious for confusing new developers. Let’s break it down together.
JavaScript is filled with amazing pre-built functions for manipulating objects and arrays. Have you yet mastered some like Array.map or Array.filter? What about the elusive Array.reduce?
Array reduce in particular can be a bit convoluted for new developers. I remember when I first started learning the array manipulator functions. Filter, sure—that made perfect sense. Map was a little more confusing, but after following through some example code it clicked. But reduce? Boi o boi. :D
Hitting up MDN, you are slapped by this description.
The reduce() method executes a reducer function (that you provide) on each element of the array, resulting in a single output value.
Moving on through the doc you are dumped into terms like accumulator without any explanation—no no. So without further ado, let’s walk through this one together with examples that make little to no sense but actually help you learn stuff.
Counting Marbles
Imagine you have a bag where you store your marbles, say const marbles = []. At first your bag is empty, but you want to store some marbles in it, so you marbles.push(3) some marbles into it. This bag has many pockets inside of it, so after adding some marbles and looking inside you see [3, 5, 1].
The objective is now to find a way to count them. We want to reduce this whole bag (array) into a single value—a count of our marbles.
When you start counting your marbles, you’ll make a pile to accumulate or group the ones you already counted. That way we can ensure we don’t count the same pocket twice.
Let’s try to put this whole explanation to code.
const marbles = [] // Our bag
marbles.push(3) // Add marbles to the inner pockets
marbles.push(5)
marbles.push(1)

console.log(marbles) // [3, 5, 1]

// How many marbles I gots?
// Count will be our group/accumulate of marbles
// The first time it runs it will be already = 3. We like being efficient here,
// so we start with the first pocket already in the count
const marbleCount = marbles.reduce((count, value, index) => {
  console.log(`Current index: ${index}`) 
  // Output 1 on first pass, then 2. Notice we skip 0 here.

  const totalSoFar = count + value
  
  console.log(`Adding ${value} marbles to our count: ${count}. Now we gots: ${totalSoFar}`)
  /*
     Fist pass: Adding 5 marbles to our count: 3. Now we gots: 8
     Second pass: Adding 1 marbles to our count: 8. Now we gots: 9
  */

  // Every time we count, we have to return
  return totalSoFar
})

console.log(`Marble count: ${marbleCount}`)
Let’s go through this slowly, because it can be a little confusing. First, we create the array that will hold our marbles, and we add them to different indexes by using array push.
We call marbles.reduce, which takes a function as a first parameter. This function takes three params: the count (also called accumulator), the value, and the index. This last one is optional.
The count will be the number of marbles we have counted SO FAR. That means it will increment each time we go through the function. It will start out with the value of marbles[0], so 3.
The value will hold the value of the next index in the loop. The reduce function will go through each one EXCEPT for index [0]—remember, we get that for free on the first count already.
The index is the current index for the array being reduced. Remember how we just mentioned that index 0 will be skipped though? This means the first pass, value will give us 5 and index will be 1. The second pass will give us a value of 1, and the index 2.
Run this code and check out the console, even play around with the console logs and the values, add more marbles to the array. Once you see it in motion, it will become clearer. Here’s a fiddle for you try it out:
https://jsfiddle.net/jx7aw3rz/4/.
Taking it a Step Further: Gemcutting 101
Ok yeah, reduce is really nice at running a fixed set of values through a simple math function, but I’m guessing you want to learn how to use it with more interesting things like processing a batch of data, or even filtering it out too.
Let’s pretend we’re building a game, some sort of RPG (Role Playing Game), and we want to create a reduce function that takes all of our player’s gems and cuts them.
Before anything though, we’re going to do a little setup with a couple of variables and a function.
const gems = [
  { 
    type: 'Ruby',
    cut: false,
    value: 100,
    requiredSkill: 1
  },
  {
    type: 'Sapphire',
    cut: false,
    value: 200,
    requiredSkill: 2
  },
  {
    type: 'Opal',
    cut: false,
    value: 250,
    requiredSkill: 8
  },
  {
    type: 'Diamond',
    cut: false,
    value: 300,
    requiredSkill: 5
  }
]

const gemcuttingSkill = 4

const cutGem = gem => {
  if (gem.requiredSkill > gemcuttingSkill) return

  gem.value = gem.value * 3
  gem.cut = true
}
Our gems array will be an array of objects, and each one will be a gem object.
Next, gemcuttingSkill will be a numeric value that will indicate the skill that the user needs to cut a gem—if the gem has a very high requiredSkill, the user won’t be able to cut it.
Finally, we have a function called cutGem. It takes a single param, a gem, and it attempts to cut it. First, however, it checks against the user’s skill. Notice also that the value of a gem increases by 3 when it’s cut!
Let’s write the reduce function that will use the cutGem function on all of our gems. We are going to make one small exception though! We will not cut any Diamonds—they’re too rare to be cut. Those go to the auction house.
const cutGems = gems.reduce((acc, gem, index) => {
  console.log('Processing gem: ', gem)
  console.log('With index', index)
  
  if (gem.type === 'Diamond') {
    // Leave Diamond uncut, and dont add it to the accumulator
    return acc
  }
  
  cutGem(gem)
  if (!gem.cut) {
    // The player wasnt able to cut it
    return acc
  }
  
  acc.push(gem)
  
  return acc
}, [])

console.log(cutGems)
We call .reduce on our gems array, and we name our accumulator parameter acc. This is a common convention that you may find in some articles and resources.
The second parameter that the function receives is the gem, because we are looping through the array.
Now I want you to take a quick look at the end of the function block, right after return acc—don’t worry, we’ll come back to the function in a minute. Do you see that empty array that we are passing as a second parameter to the reduce call? That’s the default or starting value for the accumulator.
Earlier, when we were looking at the example with the marbles, I told you that the first element had already been added to the accumulator for us for free, but what if we want to provide an initial value for our accumulator and manipulate it inside the reduce function?
The solution is to pass in a second optional argument to array.reduce that will be the default value. So the first time that the function we are providing for gems.reduce is called, acc will have a value of [], and the gem that we’re getting is going to be the one at index 0!
Ok, coming back to the contents of the function. First we are going to log a couple of things so that we can see in the output of the console how reduce is working.
Next, we add an if statement—we will check to see if the gem is of type Diamond. If it is, we will return the acc itself without modifying it, because we don’t want to add this gem to it. Keep in mind that you always have to return the acc inside the reduce function, even if you didn’t modify it.
Afterwards, we call cutGem with the current gem as a parameter. Our function will execute and cut the gem if the player has the required skill level. Then we will check to see if the gem was actually cut in the next if statement. If it hasn’t, we’re going to return the acc in this case too.
Finally, we push the cut gem into the acc array and return it.
This whole process will be executed for our bag of gems, one at a time, and we will get back into our const cutGem an array of the gems that were cut.
Let’s take it a step further and use array.reduce once again to calculate the value of the cut gems, if there are any.
const gemValue = gems.reduce((acc, gem) => {
  return acc + gem.value
}, 0)
   
console.log(`The cut gems have a value of ${gemValue} gold`)
Once again we have a reduce call on our gems array—but pay close attention to the default value. This time we’re setting it as 0.
If we skipped this tiny detail and omitted the default value for the acc, then our reduce function would receive the whole gem as the value of the acc, which is not what we want. We want to deal with numbers only.
Inside the function, we go through each cut gem, and we return the current value of the acc plus the value of the current gem. So each time we go through the loop, the gem.value will be added to the accumulated value, and so on.
Here’s a fiddle for you to execute and test this code:
https://jsfiddle.net/m7agkudx/.
Wrapping up
Array.reduce is one of those functions that has a name that sometimes isn’t very descriptive of what it does. The flexibility that we get for processing array items with this function is fantastic, but comes at a cost of a higher learning curve.
Hopefully with these couple of silly examples you are able to dive right in and start writing your own reduce functions!
As always, thanks for reading and share with me your thoughts on Twitter at: @marinamosti.
P.S. All hail the magical avocado!
P.P.S. ❤️☠️