arrow left
Back to Developer Education

    Introduction to JavaScript High Order Functions

    Introduction to JavaScript High Order Functions

    Functional programming is a significant building block for JavaScript as a language. They are some of the common scripts that you will continuously use when building JavaScript applications. Functions set statements to perform tasks. <!--more--> Functions define a task, that you can call and execute whenever you want to go inside your program.

    JavaScript treats a function as an object assigned to return a value. They have methods and properties, just like the other objects. Unlike ordinary objects, a function can be:

    • Called.
    • Passed as an argument to another function.
    • Assigned to a variable as a value.
    • Be returned by another function.

    Functions are fundamental and outside the "normal" functions, JavaScript has high order functions.

    A high order function accepts another function as an argument or returns a function as its output. The concept is commonly referred to as a callback function.

    A function that takes another function as its argument is what we refer to as a callback function because it calls back that function.

    Here is a basic example:

    // create a function
    function sayName(name, callback) {
    console.log("Hello" + " " + name);
    callback();
    }
    // callback function
    function callMe() {
    console.log("I am a callback function");
    }
    // passing function as an argument
    sayName("John Doe", callMe);
    

    Output:

    Hello John Doe
    I am a callback function
    

    In the example above, callMe() is a callback function because it's an argument of sayName(). sayName() calls it back.

    In this guide, we'll learn about the high order function. We'll discuss and show you some of the common built-in JavaScript high order functions with examples. These include filter, map, for each, and reduce.

    To follow along with this guide, prior knowledge of JavaScript functions and arrow functions will be important background knowledge to understand.

    Array.prototype.filter

    filter() allows you to filter things in an array. It's a method that returns a new array of elements. A callback function tests each element in the input array. The filter() method composes all the elements that pass the callback test and returns a new array.

    Let's digest this information with examples.

    Assume:

    1. An array of domestic animals such as:
    ["cat", "dog", "cow" ,"goat", "sheep", "donkey", "pig", "horse"]
    

    With this array we want to filter all domestic animals whose names have three-letter.

    1. An array of adult ages.
    [
      { name: "Xavier", age: 19 },
      { name: "Alice", age: 21 },
      { name: "John", age: 14 },
      { name: "David", age: 22 },
      { name: "Amanda", age: 18 },
      { name: "Amy", age: 20 },
      { name: "Jesicca", age: 28 },
      { name: "Alex", age: 15 },
    ];
    
    

    Here we want to filter adults who are 18 years and older.

    Example 0ne: without filter() method

    We'll perform a filter operation on the above arrays without using a higher-order function.

    Domestic animals

    Here we have an array Animals, and we want to filter all the animals whose names has three letters.

    const Animals = ["cat", "dog", "cow" ,"goat", "sheep", "donkey", "pig", "horse"];
    const dAnimals = [];
    // we could use a standard for loop
    for (let i = 0; i < Animals.length; i++) {
      if (Animals[i].length == 3) {
        dAnimals.push(Animals[i])
      }
    }
    console.log(dAnimals);
    
    // [ 'cat', 'dog', 'cow', 'pig' ]
    

    Adult ages

    With this example we want to filter all the adults whose age is 18 years and over.

    const adults = [
      { name: "Xavier", age: 19 },
      { name: "Alice", age: 21 },
      { name: "John", age: 14 },
      { name: "David", age: 22 },
      { name: "Amanda", age: 18 },
      { name: "Amy", age: 20 },
      { name: "Jesicca", age: 28 },
      { name: "Alex", age: 15 },
    ];
    
    
    let anAdult = [];
    for(let i = 0; i < adults.length; i++) {
     if(adults[i].age >= 18)  {
        anAdult.push(adults[i]);
      }
    }
    console.log(anAdult);
    
    /*[
      { name: 'Xavier', age: 19 },
      { name: 'Alice', age: 21 },
      { name: 'David', age: 22 },
      { name: 'Amanda', age: 18 },
      { name: 'Amy', age: 20 },
      { name: 'Jesicca', age: 28 }
    ]*/
    
    

    The examples above perform the test as we intended. However, these methods can get cumbersome when the number of elements being processed increases.

    The codes are a bit lengthy and need a lot of code setup. We can distill this further using high order functions.

    Example two: Using High Order Functions

    We will perform the filter operation using filter() as a high order function.

    Domestic animals

    const dAnimals = Animals.filter(function(animal) {
      if(animal.length == 3) {
        return true;
      }
    });
    console.log(dAnimals);
    
    // [ 'cat', 'dog', 'cow', 'pig' ]
    

    Adult ages

    const anAdult = adults.filter(function(adult) {
      if(adult.age >= 18) {
        return true;
      }
    });
    console.log(anAdult);
    
    /*[
      { name: 'Xavier', age: 19 },
      { name: 'Alice', age: 21 },
      { name: 'David', age: 22 },
      { name: 'Amanda', age: 18 },
      { name: 'Amy', age: 20 },
      { name: 'Jesicca', age: 28 }
    ]*/
    

    We get the same results. Yet this time, the code is much smaller and compact. We don't need any looping to iterate the input array.

    filter() calls the passed callback to each element being tested. filter() accepts the arguments of the callback function.

    The callback holds the value of the element being processed with the syntax below.

    let newArray = arr.filter(callback(currentValue[, index, [array]){
        //Return a new array element that passes the callback test
    }[, thisArg]);
    

    A filter callback function takes the following parameters:

    • currentValue - (the current value) holds the value of the element being processed by the callback test.
    • Index - holds the current index of the array element.
    • Array - an object of the array being tested.

    Example three: Covert the High Order Function to an Arrow Function

    With ES6, we can break this down further using the arrow function. This is an abbreviated way to write compact functions.

    Here is a basic example of an arrow function.

    Basic syntax

    (param1, paramN) => expression
    

    Basic example

    function sum(x, y) {
      return x + y;
    }
    

    We will replicate the example above using the arrow function.

    const sum = (x, y) => x + y;
    

    The arrow function uses different syntax compared to the regular function. However, they don't change the logic.

    They make the code compact and less verbose. Here is a guide that will help you understand the arrow function concept.

    Let's replicate filter() using the arrow function.

    const adults = [
      { name: "Xavier", age: 19 },
      { name: "Alice", age: 21 },
      { name: "John", age: 14 },
      { name: "David", age: 22 },
      { name: "Amanda", age: 18 },
      { name: "Amy", age: 20 },
      { name: "Jesicca", age: 28 },
      { name: "Alex", age: 15 },
    ];
    
    const anAdult = adults.filter(adults => adults.age >= 18);
    console.log(anAdult);
    
    /*[
      { name: 'Xavier', age: 19 },
      { name: 'Alice', age: 21 },
      { name: 'David', age: 22 },
      { name: 'Amanda', age: 18 },
      { name: 'Amy', age: 20 },
      { name: 'Jesicca', age: 28 }
    ]*/
    

    In just one line of code and we got the same results.

    So how does filter() magically get us to such satisfying results with just a few lines of code?

    Every element that passes the callback, filter() return a new array for these elements.

    That means, if no element passes the test, the array return value will be empty. An element that fails the callback test is skipped and is not included in the new filtered array.

    Array.prototype.reduce

    This is another built-in JavaScript high order function. It uses reducer as a callback function to execute each element of the input array.

    The callback takes two parameters, namely:

    1. Accumulator
    2. Current value

    It doesn't return a new array; instead, it uses a provided callback called reducer. When returned, it reduces the input array into a single value. The reducer callback makes this method a high order function.

    reduce() accept two input parameters:

    1. A callback function (the reducer function).
    2. Optional InitialValue.
    array.reduce(callback[, initialValue])
    

    The reducer callback function accepts two mandatory arguments:

    1. Accumulator - accumulates the previously returned value returned by the last callback invocation.
    2. Current value - the value of the current element being processed by the callback.
    array.reduce(callback( accumulator, currentValue[, index[, array]){
      // return result from executing something for accumulator or current Value
    }[, initialValue]);
    

    Let's take a look at this example, and assume an array of numbers:

    [5, 6, 7, 8, 9, 10]
    

    Imagine getting the sum of these array elements. We can use loops as we demonstrated in the filter() method, and we'll get the total sum of this array.

    However, we can opt to optimize the code with just a few lines using reduce(). This will make the code much cleaner.

    Let’s demonstrate this with examples.

    Using Loops (No High Order Function)

    const sum = [5, 6, 7, 8, 9, 10];
    let total = 0;
    for(let i = 0; i < sum.length; i++) {
      total = total + sum[i];
    }
    console.log(total);
    // we get 45
    

    In the example above, i loops through the provided array elements and returns their sum.

    Using High Order Function 'reducer'

    const sum = [5, 6, 7, 8, 9, 10];
    const total = sum.reduce((accumulator, currentValue) => {
    return accumulator + currentValue;
    });
    console.log(total);
    // we still get 45
    

    The callback function will be called five times. The accumulator will hold the last returned callback value. This value will be the next callback initial value.

    CallbackAccumulatorCurrentValueCurrentIndexreturned value
    1st call56111
    2nd call117218
    3rd call188326
    4th call269435
    5th call3510545

    reduce() will return the last invoked callback.

    In this case, 45.

    It's important to note, for the first callback invocation, the value of the accumulator is the value of the first element in the input array. The index of the first processed array element is 1, and the value 6.

    If you look back to callback syntax, the accumulator takes the first element as the InitialValue.

    If that didn't occur, reduce() would calculate the 1st element twice.

    Using the optional InitialValue argument, we can pass an InitialValue value as the second argument to reduce().

    const total = sum.reduce((accumulator, currentValue) => {
    return accumulator + currentValue;
    },10);
    console.log(total);
    //  we get 55
    

    In this case, we assigned InitialValue as 10. The callback function will be called six times. The InitialValue will be passed as the first callback.

    The accumulator will hold the last returned callback value. This value will be the InitialValue 10, that we have passed to reduce() as a second argument.

    CallbackAccumulatorCurrentValueCurrentIndexreturned value
    1st call105010
    2nd call106115
    3rd call157222
    4th call228338
    5th call389445
    6th call4510555

    reduce() will return the last invoked callback.

    In this case, 55.

    We can conclude that the first callback would take the first element's value if no initial value passed. If the InitialValue is passed, the callback will take the InitialValue as the first callback.

    Array.prototype.map

    The map() method composes a new array by returning the results of a callback function. It executes a provided callback function once for every element provided in the input array. The callback function is applied to all the elements of the input array.

    Let’s digest this with examples.

    Example one:

    Suppose we have an array of [2, 4, 89, 0.67, 7.47, 20]; we want to multiply every element, let's say by 4.

    This is how we can multiply the elements without a high order function.

    const inputArray = [2, 4, 89, 0.67, 7.47, 20];
    const newArray = [];
    for(let i = 0; i < inputArray.length; i++) {
      newArray.push(inputArray[i] * 4);
    }
    console.log(newArray);
    // new array [8, 16, 356, 2.68, 29.88, 80]
    

    We can pass a callback as an argument to map(). map() will multiply all the input elements and return a new array result.

    With just one line of code, we can use map() to get these results as shown in the example below:

    const inputArray = [2, 4, 89, 0.67, 7.47, 20];
    const newArray = inputArray.map(element => element * 4);
    console.log(newArray);
    // new array [8, 16, 356, 2.68, 29.88, 80]
    

    Example two:

    Suppose in our animals array example we want to return all the names as uppercase. map() will accept a callback as a mandatory argument and transform each element to Uppercase form. map() will take the uppercased elements and return a new array.

    const Animals = ["cat", "dog", "cow" ,"goat", "sheep", "donkey", "pig", "horse"];
    const upperCasedAnimals = Animals.map((element) => element.toUpperCase());
    console.log(upperCasedAnimals);
    // new array ['CAT', 'DOG', 'COW', 'GOAT', 'SHEEP', 'DONKEY', 'PIG', 'HORSE']
    

    In the examples above, map() returns a new array. map() builds a new array from the retuned callback values.

    For each

    It executes a provided callback function once for each element in ascending order. For each reference to a callback function once.

    For each takes one mandatory callback function. The function will be called once in each element in ascending order.

    The callback takes three Parameters:

    • Current value - a mandatory parameter that holds the current element value in the array.
    • Index - the array index of the current element array position.
    • Array - an optional parameter that holds a complete array from where for each was called.

    Syntax:

    array.forEach(callback (currentValue, index, array) {
      // statement
    }[thisArg]);
    

    A 'for' loop

    For example:

    const Animals = ["cat", "dog", "cow" ,"goat", "sheep", "donkey", "pig", "horse"];
    for (let i = 0; i < Animals.length; i++) {
        console.log(Animals[i]);
    }
    
    /*
    cat
    dog
    cow
    goat
    sheep
    donkey
    pig
    horse
    */
    

    Converting a 'for' Loop to a High Order Function using 'for each'

    const Animals = ["cat", "dog", "cow" ,"goat", "sheep", "donkey", "pig", "horse"];
    Animals.forEach((anim) =>{
      console.log(anim)
    })
    
    /*
    cat
    dog
    cow
    goat
    sheep
    donkey
    pig
    horse
    */
    

    Final notes

    JavaScript is a functional program. You may have used or applied a high-order function in a program without realizing you are using a high order function.

    High order functions are regular functions with the ability to accept other function arguments or return a function as an output.

    If you have ever written a function that takes another function as an argument:

    image title

    Or a function that returns another function:

    image title

    You have actually applied a concept of high order function.

    I hope this guide helps you understand high order functions and replicate that concept in your programs.


    Peer Review Contributions by: Lalithnarayan C

    Published on: Dec 1, 2020
    Updated on: Jul 15, 2024
    CTA

    Cloudzilla is FREE for React and Node.js projects

    Deploy GitHub projects across every major cloud in under 3 minutes. No credit card required.
    Get Started for Free