DEV Community

Samantha Ming
Samantha Ming

Posted on • Edited on

ES6 Arrow Functions Cheatsheet

Code Tidbit by SamanthaMing.com

Here’s a cheatsheet to show you the many ways to write your arrow functions.



// Explicit Return, Multi-Line
a => {
  return a
}

// Explicit Return, Single-Line
a => { return a }

// Implicit Return, Multi-line
a => (
  a
)

// Implicit Return, Single-Line
a => a

// Multiple Parameters, Parentheses Required
(a, b) => a, b


Enter fullscreen mode Exit fullscreen mode

Implicit vs Explicit Return

We have several ways of writing our arrow functions. This is because arrow functions can have either "implied return" or "explicit return" keyword.

With normal functions, if you want to return something, you have to use the return keyword. Arrow functions also have that. When you use the return keyword, it's called an explicit return. However, arrow functions up their game and allow something called implied return where the return keyword can be skipped. Let's look at some examples 🤓.

Example A: Normal Function



const sayHi = function(name) {
  return name
}


Enter fullscreen mode Exit fullscreen mode

Example B: Arrow Function with Explicit Return



// Multi-line
const sayHi = (name) => {
  return name
}

// Single-line
const sayHi = (name) => { return name }


Enter fullscreen mode Exit fullscreen mode

Example C: Arrow Function with Implicit Return



// Single-line
const sayHi = (name) => name

// Multi-line
const sayHi = (name) => (
  name
)


Enter fullscreen mode Exit fullscreen mode

Notice the difference? When you use curly braces {}, you need to explicitly state the return. However, when you don't use curly braces, the return is implied and you don't need it.

There's actually a name for this. When you use curly braces like in Example b, it's called a block body. And the syntax in Example c is called a concise body.

⭐️ Here are the rules:

  • Block body ➡️ return keyword is required
  • Concise body ➡️ return keyword is implied and not needed

Parentheses

With a normal function, we always had to use parentheses. However, with Arrow Functions, parentheses are optional if there is ONLY one parameter.

Parentheses are optional for a SINGLE parameter



// Normal Function
const numbers = function(one) {}

// Arrow Function, with parentheses
const numbers = (one) => {}

// Arrow Function, without parentheses
const numbers = one => {}


Enter fullscreen mode Exit fullscreen mode

Parentheses are required for MULTIPLE parameters



// Normal Function
const numbers = function(one, two) {}

// Arrow Function, with parentheses
const numbers = (one, two) => {}


Enter fullscreen mode Exit fullscreen mode

⚠️ Arrow Functions Gotcha: Returning Objects

Remember I mentioned about the different body types - concise body and block body. Just to quickly update you in case you skipped that section (I'm a bit sad, but not offended 😝). Block body is where you use curly braces and have an explicit return. Concise body is where you don't use curly braces, and you skip the return keyword. Alright, now you're caught up, let's get back to the gotcha 🤯

Let's purposely break our code, so you can learn your lesson lol 😂



const me = () => { name: "samantha" };

me(); // undefined 😱


Enter fullscreen mode Exit fullscreen mode

What?! Why isn't it returning my object. Don't worry, let's fix it by wrapping it in parentheses.



const me = () => ({ name: "samantha" });

me(); // { name: "samantha" } ✅


Enter fullscreen mode Exit fullscreen mode

⭐️ Here's the rule:

  • For a concise body, wrap object literal in parentheses

Resources


Thanks for reading ❤
Say Hello! Instagram | Twitter | Facebook | Medium | Blog

Top comments (20)

Collapse
 
shayd16 profile image
Shayne Darren

While I love the arrow function syntax of ES6, I feel like the implicit returns could be a bit problematic. I personally feel in large projects we need to enforce the use of body braces and explicit returns, as that is the only syntax that will work consistently, regardless of the changes made to the function(adding a parameter, changing the return type, adding 1 more line).

However, implicit returns and concise block are pretty useful in keeping the code concise with minimum boilerplate when you're chaining several methods together.

What are your thoughts on this? Have you encountered these issues?

Collapse
 
gmartigny profile image
Guillaume Martigny

Like a lots of things in JS, you can screw yourself if you're not careful with the flexible syntaxe.

ESlint (with airBnB rules) can help a lot with this. As you said, there's two (maybe three) ways to use arrow-functions. Here's the two syntaxes I enforce to myself.

1 Simple one-liner:

this.children.forEach(child => child.remove()); // Without implicit return
const translated = points.map(point => point.add(x, y)); // With implicit return

1.1 Object return:

const options = values.map(value => ({
  value: value,
  size: value.length,
}));

2 Multi-line operation:

object.data.forEach((data) => {
  test(data.value);
  test(data.size);
});
Collapse
 
qm3ster profile image
Mihail Malo

I feel like all optional symbols must be eradicated.
I'm just on the fence with returning someone else's undefined or even worse another return when you don't want a return.

as.forEach(x => map.set(a.name,a.value)) // Oh no, I am returning the map instance!
as.forEach(x => console.log(x)) // Oh no, I am returning `console.log`'s `undefined`

For the longest time, I would avoid that, adding a {}, especially if this is an otherwise available function.

const addToMap = x => { map.set(a.name,a.value) }
const log = x => { console.log(x) }

as.forEach(addToMap) // Ahh, bliss
as.forEach(log)

But then, on one cursed day, I once again stumbled upon ES6 tail call optimization spec. (Unlike most ES6, it is implemented ONLY in Safari & Mobile Safari, was REMOVED from Chrome, and is not in active development anywhere else.)

So, what is a tail call for that spec? Why, it's a function call the result of which is then returned, unconditionally. So you see my problem.
I have never had a night of good sleep since, and I am sure you can relate, since it would mean that functions like

const addToMap = x => map.set(a.name,a.value)
const log = x => console.log(x)

With the return value discarded as late as possible, are superior.

time to die

Collapse
 
joelnet profile image
JavaScript Joel

I personally feel in large projects we need to enforce the use of body braces and explicit returns, as that is the only syntax that will work consistently, regardless of the changes made to the function(adding a parameter, changing the return type, adding 1 more line).

I'm curious about your reasoning here and if you have any examples that could clear things up for me.

I don't see the syntax as being retired to project size.

Changing either the input or output of a function should be the same in both situations.

Collapse
 
islam profile image
Islam Sayed

Concise article like the concise body of arrow function. Thank you.

Collapse
 
samanthaming profile image
Samantha Ming

No problem, glad you found it helpful!

Collapse
 
qm3ster profile image
Mihail Malo
(a, b) => a, b

Oh god, what is that :v

Collapse
 
samanthaming profile image
Samantha Ming • Edited

I just want to show the parameters being used. But I can see how this is confusing. One shouldn't return with a , for sure. But for those wondering why, see below for the output of this function:

const someFunc = (a, b) => {
  return a, b
}

someFunc('a', 'b'); // 'b'
// It returns only the last argument
Collapse
 
qm3ster profile image
Mihail Malo • Edited

No, the real WUT is this:

let fn = (a, b) => a, b
fn('a','b') // It won't be 'b' :)

fn = (a, b) => (a, b)
fn('a','b') // Now it will.

// Honorable mention:
fn = ((a, b) => a, b)
fn('a','b') // Uncaught TypeError: fn is not a function
// (It's actually the `undefined` `b` declared at line 1:23)
// How excellent is that?!
Collapse
 
wolfhoundjesse profile image
Jesse M. Holmes

I've had to consider which of these I like:

// plural/singular relationships
this.children.forEach(child => child.name = 'George');
// initials, picked up from Linq usage
this.children.forEach(c => c.name = 'George');
Collapse
 
samanthaming profile image
Samantha Ming

I personally prefer the plural/singular because it’s more descriptive. It might not be as obvious because your example is only a one-liner. But once you have more logic inside the loop, it’s helpful for others to follow along your code and even yourself down the road. It takes removes the guessing 😊

Collapse
 
equiman profile image
Camilo Martinez

And what about arrow functions with zero parameters?

Collapse
 
samanthaming profile image
Samantha Ming • Edited

That should definitely be in the notes!

// zero parameters require parentheses
() => "👍"
Collapse
 
anortef profile image
Adrián Norte

I'm a simple man, I see arrow functions I click love. :P

Collapse
 
samanthaming profile image
Samantha Ming

Arrow functions FTW 😆🙌

Collapse
 
habelfoc profile image
Habel Justin

thanks for this

Collapse
 
samanthaming profile image
Samantha Ming

You're welcome! Glad you found it helpful 👍

Collapse
 
slumboy profile image
slumboy

Wowww.. I see you in Twitter and here again. __^

Collapse
 
samanthaming profile image
Samantha Ming

Yup! I post my code notes here and on medium 😄

Collapse
 
edusouzaprogrammer profile image
Edu Souza

what wonderful content!