What is reduce and why should I care?

Published by Johan Tell

Functional programming is in many cases tied to immutability and lack of for-loops that are available in the procedural paradigm. Instead we’re told to use concepts like recursion or the more high level toolkit that belongs of map, filter and similar functions.

While map for example is a very good tool which should be used as often as possible it just simply can’t do certain kinds of tasks. As an example map is strictly used for 1:1 transformations where the amount of items in the list we’re performing the map on will be the same in the result. To get around that problem we could potentially do something like this:

[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
|> Enum.map(fn (item) ->
  if item % 3 != 0 do
    item * 2
  else
    nil
  end
 end) # [1, 4, nil, 8, 10....]
|> Enum.reject(fn (item) -> item == nil end)
#=> [2, 4, 8, 10, 14, 16, 20]

This solves the problem 100% but it’s also an operation that forces us to iterate and perform another step which not only causes us to cycle the list twice, which means that for every item in the initial list, we’ll require double the amount of time. This is not an issue if we’re working with smaller datasets since twice the amount of time on almost nothing is still almost nothing.

Enter reduce

The reduce function is what we can call a lower level abstraction of most of the functions we’re using to perform transformations on a list. In fact, you can use reduce to reimplement most other functions available on lists/arrays but you can’t do it the other way around. So how does it work?!

In order to understand reduce it helps to first first think of data as immutable, once it’s defined it might not be changed.

When we’ve accepted the fact that we cannot ever change anything we’ll come to the realization that in order for us to perform transformations, our only chance is by creating new things. This is exactly what reduce is about, we’re using the knowledge of something we have, in order to create something new, one step at a time. Summation is a common example of how we can transform a list of items into something else, let’s use reduce to subtract money with multiple currencies from a bank account instead:

purchases = [
  %Money{amount: 5, currency: :SEK},
  %Money{amount: 9, currency: :USD},
  %Money{amount: 1, currency: :SEK},
  %Money{amount: 3, currency: :USD},
  %Money{amount: 7, currency: :USD},
]
bank_account = %BankAccount{
  SEK: 5000,
  USD: 127,
}

Enum.reduce(list, bank_account, fn (purchase, bank_account) ->
  IO.inspect({bank_account, purchase})

  purchase_currency = purchase.currency
  amount_after_purchase = bank_account[purchase.currency] - purchase_currency

  if amount_after_purchase < 0 do
    raise "Not enough funds on your bank account"
  end

  %BankAccount{bank_account | ^purchase_currency: amount_after_puchase}
end)

If we were to run this code we’d get logs that states the current value of the bank account in each iteration and the purchase that is going to be processed:

# Data for processing first purchase
{%BankAccount{SEK: 5000, USD: 127}, %Money{amount: 5, currency: :SEK}}

# Before processing second purchase
{%BankAccount{SEK: 4995, USD: 127}, %Money{amount: 9, currency: :USD}}
{%BankAccount{SEK: 4995, USD: 118}, %Money{amount: 1, currency: :SEK}}
{%BankAccount{SEK: 4994, USD: 118}, %Money{amount: 3, currency: :USD}}
{%BankAccount{SEK: 4994, USD: 115}, %Money{amount: 7, currency: :USD}}

# Lastly the last operation will be processed and the bank account will be
# returned since that is the value we're transforming into.
# => %BankAccount{SEK: 4994, USD: 108}

So that is how we can use reduce to do a form of summation (we’re using the bank account as the initial value and applying a list of data with the help of a reducer function).

Other use cases

reduce isn’t always about reducing a list of values into one value, it could also be about performing business logic in an efficient matter. As an example it’s quite common that we’d like to perform multiple blocks of logic onto a list of items. Let’s consider the case where we’d like to group a list of people by the first letter of their last name but only show adults in the result:

persons = [
  %Person{first_name: "Eli", last_name: "Wood", age: 31},
  %Person{first_name: "Karina", last_name: "Becker", age: 17},
  %Person{first_name: "Albert", last_name: "Horton", age: 22},
  %Person{first_name: "Margaret", last_name: "Winston", age: 67},
]

persons
Enum.reduce(%{}, fn (person, grouped_persons) ->
  if person.age =< 18 do
    grouped_persons
  else
    last_name_initial = String.at(person.last_name, 0)
    %{grouped_persons | ^last_name_initial: (grouped_persons[last_name_initial]
}} []) ++ [person]}
  end
end)

With reduce we’re able to do both things in one pass over the list instead of two, which we could have done by using filter and group_by.

Closing words

While reduce isn’t necessary in most cases (especially where performance doesn’t matter) it can for the most part be replaced with functions such as map, filter, group_by, sum, min, max. Since those are higher level and their descriptive nature makes them good options for both readability and maintainability.

However there might be cases where the problem requires a more flexible tool or where milliseconds can be the difference being a good and a bad experience. For those cases it could be good to know how you can utilize the powerful reduce to be able to reach your goals!

Johan Tell
Johan Tell