?? vs || in JavaScript: The little-known difference

At first glance you think you can just swap in anyone you like right?

JavaScript
let fastest = undefined; console.log(fastest ?? 'The Flash⚡'); // The Flash⚡ console.log(fastest || 'The Flash⚡'); // The Flash⚡

Wrong. They’re not what you think.

And we must learn the difference once and for all to avoid painful bugs down the line.

And what’s this difference?

It’s the incredible contrast in how they treat truthy and falsy values.

What are these?

Falsy: becomes false in a Boolean() or if:

  • 0
  • undefined
  • null
  • NaN
  • false
  • '' (empty string)

Truthy: every single thing else:

Now see what happens when you create a || chain like this:

JavaScript
const fruit = undefined || null || '' || 'banana🍌'; const color = '' || 'red🔴' || null; const num = 0 || NaN || 100 || false; console.log(fruit); // 'banana🍌' console.log(color); // 'red🔴 console.log(num); // 100

It keeps going until it hits the first truthy!

But what about a ?? chain? 👇

JavaScript
const fruit = undefined ?? null ?? '' ?? 'banana🍌'; const color = '' ?? 'red🔴' ?? null; const num = 0 ?? NaN ?? 100 ?? false; console.log(fruit); // '' (empty string) console.log(color); // '' console.log(num); // 0

Do you see the clear difference?

One looks for truthy, the other looks for anything that isn’t null or undefined.

When to use ?? vs ||

Initializing extra lives in a video game, where 0 means something?

?? 👇

JavaScript
const data = loadData(); const player = initPlayer(data); function initPlayer({ extraLives }) { const extraLives = extraLives ?? 20; console.log(`Player extra lives: ${extraLives}`); }

Paginating a response, where 0 limit makes no sense?

|| 👇

JavaScript
function paginate(options = {}) { return ['a', 'b', 'c', 'd', 'e'].splice(0, options.limit || 3); } paginate(1); // Output: ['a'] paginate(); // Output: ['a', 'b', 'c'] paginate(0); // Output: ['a', 'b', 'c']

User must have a name, so no spaces and definitely no empty strings?

|| 👇

JavaScript
function getUsername(userInput) { return userInput || 'Guest'; } function getProfilePic(userInput) { return userInput || 'profile-pic.png'; }

Did the user enter an invalid number or did they enter a number at all?

Find out with ?? 👇

JavaScript
function getUserAge(input) { if (!input) { return null; } else { return Number(input); } } console.log(getUserAge('') ?? '❌'); console.log(getUserAge('314') ?? '❌'); console.log(getUserAge('green🟢') ?? '❌');

?? and ?. are friends

Is someone there? not null or undefined?

Yes:

  • ?? — alright I’m done here
  • ?. — So let’s check out what you’ve gone

No:

  • ?? — So who’s next?
  • ?. — Um… bye!
JavaScript
console.log(''?.length); // 0 console.log('' ?? '❌'); // '' console.log('Tari Ibaba'?.length); // 10 console.log('Tari Ibaba' ?? '❌'); // 'Tari Ibaba' console.log(null?.length); // undefined console.log(null ?? '❌'); // '❌' console.log(undefined?.length); // undefined console.log(undefined ?? '❌'); // '❌'

Final thoughts

?? is a gullible child who will believe anything.

|| is a detective in search of the truthy and nothing but the truthy.



Every Crazy Thing JavaScript Does

A captivating guide to the subtle caveats and lesser-known parts of JavaScript.

Every Crazy Thing JavaScript Does

Sign up and receive a free copy immediately.

Leave a Comment

Your email address will not be published. Required fields are marked *