- Explain when we might want to use our custom
forEach
- Practice using functions as first-class objects
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.
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:let i = 0, l = dogs.length
i < l
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!
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!
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!