Do you remember something like this?
var text;
for (i = 0; i < 5; i++) {
text += "The number is " + i + "<br>";
}
For those of you completely new to using JavaScript, let’s break down that code spippet fellas.
- All that statement is saying is that – “i is initially equals to zero. For as long as i is less than five, run the code inside the code block (i.e update the ‘text’ variable) and increment i by one (signified with the ++).
This used to be the standard way of looping through an array of items.
Used to.
EmcaScript 2016 (ES6) came with a set of utility methods for looping through an array. For those of you wondering, EmcaScript is a general-purpose programming language, standardised by Ecma International according to the document ECMA-262.
I like to think of EmcaScript as a new, modern and improved way of writing JavaScript.
N/B: If you’re looking for a more in-depth guide on ES6 syntax, I’ll recommend HTML To React By Sleepless Yogi . It’s really good!
These array iterators includes:
- Array.map()
- Array.find()
- Array.filter()
- Array.forEach()
- Array.every()
- Array.some()
- Array.sort()
- Array.reduce()
To properly understand how these methods work, we have to first understand two concepts:
- The size of an array
- The shape of an array
Consider this code:
const team = [
{
name: 'jack',
position: 'backend engineer'
},
{
name: 'lara',
position: 'ux designer'
},
{
name: 'kingsley',
position: 'developer'
},
{
name: 'mark',
position: 'manager'
},
{
name: 'chris',
position: 'developer'
}
]
javascript
There are two thing to note about this array:
- It has a size of five items (objects),
- Each item has a shape: a name and position.
With these in mind, all of the array iterator methods has it’s own behaviour:
-
map()
- Behaviour: Changes the shape, but not the size,
- Example: Get just the name of everyone on the team.
team.map((member) => {
return member.name;
})
//Result: ['jack', 'lara', 'kingsley', 'mark', 'chris']
sort()
- Behaviour: Changes neither the size nor the shape, but changes the order.
- Example: Get the team members in alphabetical order.
team.sort();
//Result: [{name: 'chris', position: 'developer'}, {name: 'jack' ...}]
filter()
- Behaviour: Changes the size, but not the shape.
- Example: Get only the developers.
team.filter((member) => {
return member.position = “developer”;
})
// Result: [{name: 'kingsley', position: 'developer'}, {name: 'chris', position:'developer'}]
find()
- Behavior: Changes the size to exactly one, but not the shape. It Does not return an array.
- Example: Get the manager.
team.find((member) => {
return member.position = manager;
})
// Result: {name: 'justin', position: 'manager'}
N/B: If there were two managers present, find() will only return the first
forEach()
- Behaviour: Uses the shape, but returns nothing.
- Example: Give all members a bonus!
Function sendBonus(member) {
//code for bonus goes here\
}
team.forEach((member) => {
sendBonus(member);
})
//Result: Jack gets a bonus! Lara get a bonus!… (but no return value).
reduce()
- Action: Changes the size and the shape to pretty much anything you want.
- Example: Get the unique team positions.
const uniquePos = team.reduce((uniquePos, member) => {
if (uniquePos.includes(member.position)) {
return uniquePos;
}
return [...uniquePos, member.position];
}, []);
// Result: [‘backend engineer’, ‘ux designer’, ‘developer’, ‘manager’]
A bit confused, let’s break down this code guys.
The essence of this all is to get every unique postion of each member. Duplicate positions (developer) will be picked once.
The first parameter taken by the reduce method is the ‘collectedValue’from the last iteration. When the iteration is just starting, then the reduce() second argument ([] in our case) will be used. For every iteration, the collected or total value is added the current item.
In our case, when iteration just starts, the initial value [] wraps the current item in the array.
The conditional checks to see if the total value (the array at this point in the loop) includes the position of the current member. If it does, the position is ignored and the CollectedValue is returned as it was. If not, the position of the current member is added to the array (with the spread operator).
every()
- Behavior: Changes neither the size nor the shape. Returns a Boolean: true if all items meet a condition, false if any doesn’t.
- Example: Check if all items are objects.
team.every((member) => {
return typeof member = object;
})
// Results: True
some()
- Behavior: Changes neither the size nor the shape. Returns a Boolean: true if any of the items meet a condition, false if all doesn’t.
- Example: Check if any of the items is number.
team.some((member) => {
return typeof member = number;
})
// Result: False
That’s it. I hope you learnt something from these brothers and sisters. If you did, consider buying me my favourite fruit:
I will really appreciate it.
Thank you and see you soon.
Top comments (4)
I would suggest not ditch
for
and use it appropriately, e.g. for asynchronous operations and iterators.For loops are actually much more flexible, and the JS parser is faster to interpret them in some benchmarks.
Let's look at a simple example where for loops beat array methods.
Say you have an array of numbers but some elements are null, e.g. [1, null, 3, null], now you want to transform this so you remove the nulls and multiply the numbers by 2.
With array methods you'd filter, then map. That's already 2 loops.
Just write 1 for loop that continues when it sees a null, and pushes 2*number into a new array.
Don't use the wrong tools for the job.
This is what I was thinking. It isn't the point of the syntax to be cleaner or replace for... loops. It's to provide an alternative when appropriate. Just like you said use the right tool :)
Nice methods for working with arrays. Can't imagine not using array.filter() or array.find()