Skip to content

Latest commit

 

History

History
312 lines (217 loc) · 10.5 KB

higher-order-functions.md

File metadata and controls

312 lines (217 loc) · 10.5 KB

JavaScript Higher-Order Functions

Overview

Higher-order functions are built in methods in JavaScript. They are a specific subset of built in methods that take a function as an argument.

Previous Lecture (1 hour 40 min)

YouTube

Learning Objectives

  • can define .map() and .filter() as array methods
  • can distinguish between a regular function and a higher-order function
  • can distinguish between a regular built in method and a higher-order function
  • can utilize the required parameter for .map() and .filter()
  • can recall the optional parameters for .map() and .filter()
  • can demonstrate the proper use of both .map() and .filter() to produce expected outcomes

Vocabulary

  • higher-order function
  • .map()
  • .filter()
  • type coercion
  • falsy
  • truthy

Process

  • cd into the javascript-foundations-challenges repository
  • Create a new branch: higher-order-initials1-initials2 (ex. higher-order-aw-sp)
  • touch a file with no spaces and .js extension: higher-order-student1-student2.js (ex. higher-order-austin-sarah.js)
  • Open the folder in a text editor
  • Code!

Troubleshooting Tips

  • Does the function have a return?
  • Is there a console.log on the function invocation?
  • Does the number of arguments match the number of parameters?

Higher-Order Functions

A higher-order function is a built in method in JavaScript that takes in a function as an argument. Often higher-order functions are designed to operate on arrays. The function passed as an argument will execute once for every item in the array. This makes higher-order functions incredible useful for iteration.

We will focus on two higher-order functions for this lesson .map() and .filter(). Both of these have three built-in parameters: the current value, the index, and the array the higher-order function is called on.

What the parameters are named is technically arbitrary, but the order will always be the same. The higher-order functions have to take at least one argument (value) but can take two or three if needed.

.map(value, index, array)
.filter(value, index, array)

  1. The value of the array at the current iteration (required).

  2. The index of the array at the current iteration (optional).

  3. The array the higher-order function was called on (optional).

Map

The .map() method iterates through an array and returns a new array of the same length.

Here is an example of iterating using a for loop:

const myArray = [5, 3, 2, 9, 8]

let newArray = []
for (let i = 0; i < myArray.length; i++) {
  newArray.push(myArray[i] * 3)
}

console.log(newArray)
// output: [15, 9, 6, 27, 24]

Now let's look at this same objective with .map() method.

const myArray = [5, 3, 2, 9, 8]

const multiplier = myArray.map((value) => {
  return value * 3
})

console.log(multiplier)
// output: [15, 9, 6, 27, 24]

Because we are using an arrow function we can reduce the code even further. If there is only one line of code to be executed, we can remove the return keyword as the return is implied.

const myArray = [5, 3, 2, 9, 8]

const multiplier = myArray.map((value) => value * 3)

console.log(multiplier)
// output: [15, 9, 6, 27, 24]

And finally let's wrap the .map() logic in a function that can take in any array to make the function reusable. This allows us to call the multiplier function as many times as we deem useful.

const myArray = [5, 3, 2, 9, 8]

const multiplier = (array) => {
  return array.map((value) => value * 3)
}
console.log(multiplier(myArray))
// output: [15, 9, 6, 27, 24]

console.log(multiplier([3, 4, 5, 6, 7]))
// output:[9, 12, 15, 18, 21]

Notice that .map() iterates through the specified array AND returns a new array with the updated values.

Here is another example with proper pseudo coding:

// create a function that takes in an array of words and returns array with all the words capitalized
const fruitArray = ["banana", "mango", "apple", "grape"]

// declare a function that takes in an array
const capitalizer = (array) => {
  // return the array output from the map action
  return array.map((value) => {
    // with each value access the first index and capitalize the letter then concatenate the remainder of the word
    return value[0].toUpperCase() + value.substring(1)
  })
}
// logging the function and pass in the array
console.log(capitalizer(fruitArray))
// output: ["Banana", "Mango", "Apple", "Grape"]

Note: - This function has a return for the outer function AND a return for the higher-order function.

Truthy and Falsy

The JavaScript language supports type coercion. Type coercion is the evaluation of data that are not the same type by the conversion of one data type to another. Part of type coercion, and a quirk of JavaScript, is that all data types evaluate to either a true value or a false one. A falsy value is one that JavaScript deems to equate to the Boolean value false through the use of type coercion such as 0, null, and strings with no characters. A truthy value is one that JavaScript deems to equate to the Boolean value true through the use of type coercion such as non-zero numbers and strings with characters. This produces interesting outputs such as the following:

1 + "1"
// output: "11"

true + 1
// output: 2

null + 4
// output: 4

Truthy and falsy values are important to understand and can be used to the advantage of the JavaScript developer. However, this quirk of JavaScript should not be exploited at the cost of code quality.

Filter

The .filter() method iterates through an array and returns a new array with only the values that satisfy the stated condition. Filter needs an evaluation. The evaluation can be anything that will return a Boolean value of true or false or a truthy or falsy value. Values that are true or truthy are included in the subset while values that are false or falsy are excluded. This evaluation process allows filter to return a subset of the original array.

Here is an example of iterating and returning a subset of an array using a for loop:

const numbersArray = [1, 2, 7, 4, 10, 8, 9]

const onlyEven = (array) => {
  let newArr = []
  for (let i = 0; i < array.length; i++) {
    if (array[i] % 2 === 0) {
      newArr.push(array[i])
    }
  }
  return newArr
}

console.log(onlyEven(numbersArray))
// output: [2, 4, 10, 8]

In the above example, we have created a function that takes an array, loops through the array, and returns a new array containing only the even numbers.

Here is the same example using the filter method.

const numbersArray = [1, 2, 7, 4, 10, 8, 9]

const onlyEven = (array) => {
  return array.filter((value) => value % 2 === 0)
}

console.log(onlyEven(numbersArray))
// output: [2, 4, 10, 8]

Here is another example with proper pseudo coding:

// create a function that returns only the values at the odd indexes
const numbers = [4, 24, 5, 9, 0, 78]

// declare a function that takes in an array
const getOnlyOddIndex = (array) => {
  // return a higher-order function that takes in the value and the index
  return array.filter((value, index) => {
    // return only the values that have an odd index
    return index % 2 !== 0
  })
}
// log the function and pass in the array
console.log(getOnlyOddIndex(numbers))
// output: [24, 9, 78]

Map vs Filter

When deciding which higher-order function to use, remember these two differences:

  • .map() will do something to each item in an array and return an array of the same length.
  • .filter() will make a decision about each item in an array and return a subset based on the criteria given.

💻 Challenges

Copy the challenges into your JavaScript file. Comment out the instructions and code the solution to each problem beneath the prompt.

  • Create a function that takes in an array of numbers and returns an array with all numbers multiplied by 10.
const arr1 = [3, 9, 15, 4, 10]
// output: [30, 90, 150, 40, 100]
  • Create a function that takes in an array of numbers and returns an array with all numbers divided by two.
const arr1 = [3, 9, 15, 4, 10]
// output: [1.5, 4.5, 7.5, 2, 5]
  • Create a function that takes in an array of numbers and returns an array with only odd numbers.
const arr2 = [2, 7, 3, 5, 8, 10, 13, -9]
// output: [7, 3, 5, 13, -9]
  • Create a function that takes in a string of multiple words and returns an array with only the words that have an odd number of characters.
const pumbaa =
  "Hakuna Matata what a wonderful phrase Hakuna Matata ain't no passing craze"
// output: ["a", "wonderful", "ain't", "passing", "craze"]
  • Create a function that takes in an array of numbers and letters and returns a string with only the letters. HINT: Use the typeof operator.
const comboArr = [7, "n", "i", "c", 10, "e", false, "w", 3, "o", "r", "k"]
// output: "nicework"
  • Create a function that takes in an array and returns an array without any false, null, 0, or blank values.
const filterArrayValues = [58, "", "abcd", true, null, false, 0]
// output: [58, "abcd", true]
  • Create a function that takes in an array of strings and returns an array of strings with every other letter capitalized.
const makesWackyWords = ["simba", "nala", "zazu", "rafiki"]
// output: ["sImBa", "nAlA", "zAzU", "rAfIkI"]
  • Create a function that takes in a string and returns a new string with all the vowels removed.
const str = "javascript is awesome"
// output: "jvscrpt s wsm"
  • Create a function that takes in a string containing numbers and returns an array with each number at its own index and converted to the data type of number.
const stringOfNumbers = "4574328"
// output: [4, 5, 7, 4, 3, 2, 8]

🏔 Stretch Goals

  • Create a function that takes in two arrays as arguments returns one array with no duplicate values.
const arr1 = [3, 7, 10, 5, 4, 3]
const arr2 = [7, 8, 2, 1, 5, 4]
// output: [3, 7, 10, 5, 4, 8, 2, 1]
  • Create a function that takes in an array of mixed data types and returns the first value that is a string. HINT: look into the JavaScript .find() method.
const allTheData = [7, null, true, 2, "yo!", false, 4, "hello!"]
// output: "yo!"

Back to Syllabus