forEach Code-Along

Objectives

  1. Explain when we might want to use our custom forEach
  2. Practice using functions as first-class objects

Why use forEach?

forEach provides a terse way to iterate over elements in an array, but it doesn't return anything on its own. Instead, it changes the objects in place and always returns undefined. As such, forEach is most useful when we want to execute code that has side effects on the elements that we're operating on.

Additionally, forEach does not work with objects. So even though JavaScript lets us iterate over the key-value pairs in objects using for loops, we don't have a ready-to-hand way of doing so with the terseness of forEach.

Well, right now, we don't — by the end of this lesson, we'll have learned how to write a utility function that can iterate over arrays as well as objects.

Objects vs. Arrays — Refresher

Before we dive in, let's go through a quick review of what it looks like to iterate over an array versus what it looks like to iterate over an object. When we iterate over an array, we access each element according to its index.

Input the following in your console. What does it print?

const dogs = [
  { name: "Fido", age: 2 },
  { name: "Odie", age: 8 },
  { name: "Ralph", age: 4 }
];

for (let i = 0, l = dogs.length; i < l; i++) {
  console.log(i);
}

You should see:

0
1
2

Now enter the above again, but change the log statement to console.log(dogs[i]). This should result in the following output:

{name: "Fido", age: 2}
{name: "Odie", age: 8}
{name: "Ralph", age: 4}

Which is great! But maybe we also want to take a closer look at each dog. Let's take a step back and focus on Fido for now. Run the following code:

const dog = { name: "Fido", age: 2 };

for (const key in dog) {
  console.log(key);
}

What you should see is:

name
age

This clearly illustrates that we're looping over the dog object's properties, and we're logging out the key of the key-value pair for each property. Now, let's access the value of the properties by changing the log statement to console.log(dog[key]):

Fido
2

So, what are the differences here between accessing elements in an array and accessing key-value pairs in an object?

  • In an array, we most often iterate over the indexes of the array starting at index 0 and going through all the elements.
  • In an object, we most often iterate over its keys. These are unordered, so we just go through all of them.
  • In an array, we use a standard for loop with three clauses:
    1. let i = 0, l = dogs.length
    2. i < l
    3. i++
  • In an object, we can use a for...in loop to shorten our keystrokes a little: for (const key in dog) { }, as seen in the example above.

Okay, after that refresher, let's dive into some code!

Each and Every One

Let's start by defining our function signature — we know that it's going to take something that we can iterate over (the iterable), and a function to iterate with (the callback). The thing that we iterate over can be either an array or an object, so we want to be generic with how we name it. In other languages, we could specify a type; but JavaScript does not allow us such safety; so the best we can do is name things precisely and type-check on the fly.

function forEach(iterable, callback) {}

Note that forEach does not conflict with Array.prototype.forEach because the latter is just a property of Array.prototype — it is not a global function.

Now, inside our forEach function, we want to do one thing if iterable is an array, and another thing if iterable is an object. But here's the catch: we can't just use the typeof operator, because typeof [] is 'object'. That's not very helpful, since that provides us with no way to tell the difference between an array and an object...

Thankfully, JavaScript has thought of this. We can use a method attached to Array, called — wait for it — isArray(). We use it like so:

Array.isArray([]); // true
Array.isArray(1);  // false

So let's dive into our function and first check if iterable is an array.

function forEach(iterable, callback) {
  if (Array.isArray(iterable)) {

  }
}

If iterable is an array, we want our function to behave like Array.prototype.forEach. Recall that the built-in forEach passes the current element, the element's index, and the whole array to the callback. Since we're replicating that functionality, we should do the same.

function forEach(iterable, callback) {
  if (Array.isArray(iterable)) {
    for (let i = 0, l = iterable.length; i < l; i++) {
      const element = iterable[i];
      callback(element, i, iterable);
    }
  }
}

The signature of our callback might not specify all three arguments, but that's okay! We leave it up to the implementation of callback to decide what arguments it cares about. Inside our forEach, we simply pass along everything that's relevant.

Now that we've handled the array part, we can check if iterable is an object. To do so, we can use typeof here — if our code has gotten this far, we know that iterable is not an array and won't give a false positive.

function forEach(iterable, callback) {
  if (Array.isArray(iterable)) {
    for (let i = 0, l = iterable.length; i < l; i++) {
      const element = iterable[i];
      callback(element, i, iterable);
    }
  } else if (typeof iterable === 'object') {

  }
}

So what do we do inside that else if block? Well, we iterate the object, just like we did before:

for (const key in iterable) {
  const value = iterable[key];
  callback(value, key, iterable);
}

Just as in the array loop, we pass the element, the key of the current property (remember, the "keys" in an array are just the indexes), and the whole iterable. So now our whole function looks like this:

function forEach(iterable, callback) {
  if (Array.isArray(iterable)) {
    for (let i = 0, l = iterable.length; i < l; i++) {
      const element = iterable[i];
      callback(element, i, iterable);
    }
  } else if (typeof iterable === 'object') {
    for (const key in iterable) {
      const value = iterable[key];
      callback(value, key, iterable);
    }
  }
}

There's a minor problem here, but we'll return to it at a later date. (If you want to jump ahead a bit, read more about Object.prototype.hasOwnProperty.) We can, at this point, just say —

We did it!

yay!

You might ask why we don't just simplify our checks into something a lot simpler:

for (const keyOrIndex in iterable) {
  callback(iterable[keyOrIndex], keyOrIndex, iterable);
}

We could do that, and it would, for the most part, work. The trouble is, we don't set properties of arrays the same way we set properties of objects. So if we attempted to iterate over an array with for...in, but had only explicitly set some distant index, we would only see the value for that index. This StackOverflow post provides a few examples for why this is a bad idea.

Lastly, run learn submit. Your tests should pass and you should be ready to go! Well done!