DEV Community

Artem Sapegin
Artem Sapegin

Posted on • Originally published at sapegin.me

Washing your code: don’t make me think

You’re reading an excerpt from my book on clean code, “Washing your code.” Available as PDF, EPUB, and as a paperback and Kindle edition. Get your copy now.


Clever code is something we may see in job interview questions or language quizzes, when they expect us to know how a language feature, which we probably have never seen before, works. My answer to all these questions is: “it won’t pass code review”.

Some folks confuse brevity with clarity. Short code (brevity) isn’t always the clearest code (clarity), often it’s the opposite. Striving to make your code shorter is a noble goal, but it should never come at the expense of readability.

There are many ways to express the same idea in code, and some are easier to understand than others. We should always aim to reduce the cognitive load of the next developer who reads our code. Every time we stumble on something that isn’t immediately obvious, we waste our brain’s resources.

Info: I “stole” the name of this chapter from Steve Krug’s book on web usability with the same name.

Dark patterns of JavaScript

Let’s look at some examples. Try to cover the answers and guess what these code snippets do. Then, count how many you guessed correctly.

Example 1:

const percent = 5;
const percentString = percent.toString().concat('%');
Enter fullscreen mode Exit fullscreen mode

This code only adds the % sign to a number and should be rewritten as:

const percent = 5;
const percentString = `${percent}%`;
// → '5%'
Enter fullscreen mode Exit fullscreen mode

Example 2:

const url = 'index.html?id=5';
if (~url.indexOf('id')) {
  // Something fishy here…
}
Enter fullscreen mode Exit fullscreen mode

The ~ symbol is called the bitwise NOT operator. Its useful effect here is that it returns a falsy value only when indexOf() returns -1. This code should be rewritten as:

const url = 'index.html?id=5';
if (url.includes('id')) {
  // Something fishy here…
}
Enter fullscreen mode Exit fullscreen mode

Example 3:

const value = ~~3.14;
Enter fullscreen mode Exit fullscreen mode

Another obscure use of the bitwise NOT operator is to discard the fractional portion of a number. Use Math.trunc() instead:

const value = Math.trunc(3.14);
// → 3
Enter fullscreen mode Exit fullscreen mode

Example 4:

if (dogs.length + cats.length > 0) {
  // Something fishy here…
}
Enter fullscreen mode Exit fullscreen mode

This one is understandable after a moment: it checks if either of the two arrays has any elements. However, it’s better to make it clearer:

if (dogs.length > 0 || cats.length > 0) {
  // Something fishy here…
}
Enter fullscreen mode Exit fullscreen mode

Example 5:

const header = 'filename="pizza.rar"';
const filename = header.split('filename=')[1].slice(1, -1);
Enter fullscreen mode Exit fullscreen mode

This one took me a while to understand. Imagine we have a portion of a URL, such as filename="pizza". First, we split the string by = and take the second part, "pizza". Then, we slice the first and the last characters to get pizza.

I’d likely use a regular expression here:

const header = 'filename="pizza.rar"';
const filename = header.match(/filename="(.*?)"/)[1];
// → 'pizza'
Enter fullscreen mode Exit fullscreen mode

Or, even better, the URLSearchParams API:

const header = 'filename="pizza.rar"';
const filename = new URLSearchParams(header)
  .get('filename')
  .replaceAll(/^"|"$/g, '');
// → 'pizza'
Enter fullscreen mode Exit fullscreen mode

These quotes are weird, though. Normally we don’t need quotes around URL parameters, so talking to the backend developer could be a good idea.

Example 6:

const condition = true;
const object = {
  ...(condition && { value: 42 })
};
Enter fullscreen mode Exit fullscreen mode

In the code above, we add a property to an object when the condition is true, otherwise we do nothing. The intention is more obvious when we explicitly define objects to destructure rather than relying on destructuring of falsy values:

const condition = true;
const object = {
  ...(condition ? { value: 42 } : {})
};
// → { value: 42 }
Enter fullscreen mode Exit fullscreen mode

I usually prefer when objects don’t change shape, so I’d move the condition inside the value field:

const condition = true;
const object = {
  value: condition ? 42 : undefined
};
// → { value: 42 }
Enter fullscreen mode Exit fullscreen mode

Example 7:

const array = [...Array(10).keys()];
Enter fullscreen mode Exit fullscreen mode

This wonderful one-liner creates an array filled with numbers from 0 to 9. Array(10) creates an array with 10 empty elements, then the keys() method returns the keys (numbers from 0 to 9) as an iterator, which we then convert into a plain array using the spread syntax. Exploding head emoji…

We can rewrite it using a for loop:

const array = [];
for (let i = 0; i < 10; i++) {
  array.push(i);
}
// → [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
Enter fullscreen mode Exit fullscreen mode

As much as I like to avoid loops in my code, the loop version is more readable for me.

Somewhere in the middle would be using the Array.from() method:

const array = Array.from({ length: 10 }).map((_, i) => i);
// → [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
Enter fullscreen mode Exit fullscreen mode

The Array.from({length: 10}) creates an array with 10 undefined elements, then using the map() method, we fill the array with numbers from 0 to 9.

We can write it shorter by using Array.from()’s map callback:

const array = Array.from({ length: 10 }, (_, i) => i);
// → [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
Enter fullscreen mode Exit fullscreen mode

Explicit map() is slightly more readable, and we don’t need to remember what the second argument of Array.from() does. Additionally, Array.from({length: 10}) is slightly more readable than Array(10). Though only slightly.

So, what’s your score? I think mine would be around 3/7.

Gray areas

Some patterns tread the line between cleverness and readability.

For example, using Boolean to filter out falsy array elements (null and 0 in this example):

const words = ['Not', null, 'enough', 0, 'cheese'].filter(Boolean);
// → ['Not', 'enough', 'cheese']
Enter fullscreen mode Exit fullscreen mode

I find this pattern acceptable; though it requires learning, it’s better than the alternative:

const words = ['Not', null, 'enough', 0, 'cheese'].filter(
  item => !!item
);
// → ['Not', 'enough', 'cheese']
Enter fullscreen mode Exit fullscreen mode

However, keep in mind that both variations filter out falsy values, so if zeros or empty strings are important, we need to explicitly filter for undefined or null:

const words = ['Not', null, 'enough', 0, 'cheese'].filter(
  item => item != null
);
// → ['Not', 'enough', 0, 'cheese']
Enter fullscreen mode Exit fullscreen mode

Make differences in code obvious

When I see two lines of tricky code that appear identical, I assume they differ in some way, but I don’t see the difference yet. Otherwise, a programmer would likely create a variable or a function for the repeated code instead of copypasting it.

For example, we have a code that generates test IDs for two different tools we use on a project, Enzyme and Codeception:

const props = {
  'data-enzyme-id': columnName
    ? `${type}-${toTitleCase(columnName)}-${rowIndex}`
    : null,
  'data-codeception-id': columnName
    ? `${type}-${toTitleCase(columnName)}-${rowIndex}`
    : null
};
// → {
//     'data-enzyme-id': 'type-Col-2',
//     'data-codeception-id': 'type-Col-2'
// }
Enter fullscreen mode Exit fullscreen mode

It’s difficult to immediately spot any differences between these two lines of code. Remember those pairs of pictures where you had to find ten differences? That’s what this code does to the reader.

While I’m generally skeptical about extreme code DRYing, this is a good case for it.

Info: We talk more about the Don’t repeat yourself principle in the Divide and conquer, or merge and relax chapter.

const testId = columnName
  ? `${type}-${toTitleCase(columnName)}-${rowIndex}`
  : null;
const props = {
  'data-enzyme-id': testId,
  'data-codeception-id': testId
};
// → {
//     'data-enzyme-id': 'type-Col-2',
//     'data-codeception-id': 'type-Col-2'
// }
Enter fullscreen mode Exit fullscreen mode

Now, there’s no doubt that the code for both test IDs is exactly the same.

Let’s look at a trickier example. Suppose we use different naming conventions for each testing tool:

const props = {
  'data-enzyme-id': columnName
    ? `${type}-${toTitleCase(columnName)}-${rowIndex}`
    : null,
  'data-codeception-id': columnName
    ? `${type}_${toTitleCase(columnName)}_${rowIndex}`
    : null
};
// → {
//     'data-enzyme-id': 'type-Col-2',
//     'data-codeception-id': 'type_Col_2'
// }
Enter fullscreen mode Exit fullscreen mode

The difference between these two lines of code is hard to notice, and we can never be sure that the name separator (- or _) is the only difference here.

In a project with such a requirement, this pattern will likely appear in many places. One way to improve it is to create functions that generate test IDs for each tool:

const joinEnzymeId = (...parts) => parts.join('-');
const joinCodeceptionId = (...parts) => parts.join('_');
const props = {
  'data-enzyme-id': columnName
    ? joinEnzymeId(type, toTitleCase(columnName), rowIndex)
    : null,
  'data-codeception-id': columnName
    ? joinCodeceptionId(type, toTitleCase(columnName), rowIndex)
    : null
};
Enter fullscreen mode Exit fullscreen mode

This is already much better, but it’s not perfect yet — the repeated code is still too large. Let’s fix this too:

const joinEnzymeId = (...parts) => parts.join('-');
const joinCodeceptionId = (...parts) => parts.join('_');
const getTestIdProps = (...parts) => ({
  'data-enzyme-id': joinEnzymeId(...parts),
  'data-codeception-id': joinCodeceptionId(...parts)
});
const props = columnName
  ? getTestIdProps(type, toTitleCase(columnName), rowIndex)
  : {};
Enter fullscreen mode Exit fullscreen mode

This is an extreme case of using small functions, and I generally try to avoid splitting code this much. However, in this case, it works well, especially if there are already many places in the project where we can use the new getTestIdProps() function.

Sometimes, code that looks nearly identical has subtle differences:

function handleSomething(documentId) {
  if (documentId) {
    dispatch(changeIsWordDocumentExportSuccessful(true));
    return;
  }
  dispatch(changeIsWordDocumentExportSuccessful(false));
}
Enter fullscreen mode Exit fullscreen mode

The only difference here is the parameter we pass to the function with a very long name. We can move the condition inside the function call:

function handleSomething(documentId) {
  dispatch(
    changeIsWordDocumentExportSuccessful(documentId !== undefined)
  );
}
Enter fullscreen mode Exit fullscreen mode

This eliminates the similar code, making the entire snippet shorter and easier to understand.

Whenever we encounter a condition that makes code slightly different, we should ask ourselves: is this condition truly necessary? If the answer is “yes”, we should ask ourselves again. Often, there’s no real need for a particular condition. For example, why do we even need to add test IDs for different tools separately? Can’t we configure one of the tools to use test IDs of the other? If we dig deep enough, we might find that no one knows the answer, or that the original reason is no longer relevant.

Consider this example:

function getAssetDirs(config) {
  return config.assetsDir
    ? Array.isArray(config.assetsDir)
      ? config.assetsDir.map(dir => ({ from: dir }))
      : [{ from: config.assetsDir }]
    : [];
}
Enter fullscreen mode Exit fullscreen mode

This code handles two edge cases: when assetsDir doesn’t exist, and when assetsDir isn’t an array. Also, the object generation code is duplicated. (And let’s not talk about nested ternaries…) We can get rid of the duplication and at least one condition:

function getAssetDirs(config) {
  const assetDirectories = config.assetsDir
    ? _.castArray(config.assetsDir)
    : [];
  return assetDirectories.map(from => ({
    from
  }));
}
Enter fullscreen mode Exit fullscreen mode

I don’t like that Lodash’s castArray() method wraps undefined in an array, which isn’t what I’d expect, but still, the result is simpler.

Avoid shortcuts

CSS has shorthand properties, and developers often overuse them. The idea is that a single property can define multiple properties at the same time. Here’s a good example:

.block {
  margin: 1rem;
}
Enter fullscreen mode Exit fullscreen mode

Which is the same as:

.block {
  margin-top: 1rem;
  margin-right: 1rem;
  margin-bottom: 1rem;
  margin-left: 1rem;
}
Enter fullscreen mode Exit fullscreen mode

One line of code instead of four, and it’s still clear what’s happening: we set the same margin on all four sides of an element.

Now, look at this example:

.block-1 {
  margin: 1rem 2rem 3rem 4rem;
}
.block-3 {
  margin: 1rem 2rem 3rem;
}
.block-2 {
  margin: 1rem 2rem;
}
Enter fullscreen mode Exit fullscreen mode

To understand what they do, we need to know that:

  • when the margin property has four values, the order is top, right, bottom, left;
  • when it has three values, the order is top, left/right, bottom;
  • and when it has two values, the order is top/bottom, left/right.

This creates an unnecessary cognitive load, and makes code harder to read, edit, and review. I avoid such shorthands.

Another issue with shorthand properties is that they can set values for properties we didn’t intend to change. Consider this example:

.block {
  font: italic bold 2rem Helvetica;
}
Enter fullscreen mode Exit fullscreen mode

This declaration sets the Helvetica font family, the font size of 2rem, and makes the text italic and bold. What we don’t see here is that it also changes the line height to the default value of normal.

My rule of thumb is to use shorthand properties only when setting a single value; otherwise, I prefer longhand properties.

Here are some good examples:

.block {
  /* Set margin on all four sides */
  margin: 1rem;

  /* Set top/bottom and left/right margins */
  margin-block: 1rem;
  margin-inline: 2rem;

  /* Set border radius to all four corners */
  border-radius: 0.5rem;

  /* Set border-width, border-style and border-color
   * This is a bit of an outlier but it’s very common and
   * it’s hard to misinterpret it because all values have
   * different types */
  border: 1px solid #c0ffee;

  /* Set top, right, bottom, and left position */
  inset: 0;
}
Enter fullscreen mode Exit fullscreen mode

And here are some examples to avoid:

.block {
  /* Set top/bottom and left/right margin */
  margin: 1rem 2rem;

  /* Set border radius to top-left/bottom-right,
   * and top-right/bottom-left corners */
  border-radius: 1em 2em;
  /* Set border radius to top-left, top-right/bottom-left,
   * and bottom-right corners */
  border-radius: 1em 2em 3em;
  /* Set border radius to top-left, top-right, bottom-right,
   * and bottom-left corners */
  border-radius: 1em 2em 3em 4em;

  /* Set background-color, background-image, background-repeat,
   * and background-position */
  background: #bada55 url(images/tacocat.gif) no-repeat left top;

  /* Set top, right, bottom, and left */
  inset: 0 20px 0 20px;
}
Enter fullscreen mode Exit fullscreen mode

While shorthand properties indeed make the code shorter, they often make it significantly harder to read, so use them with caution.

Write parallel code

Eliminating conditions isn’t always possible. However, there are ways to make differences in code branches easier to spot. One of my favorite approaches is what I call parallel coding.

Consider this example:

function RecipeName({ name, subrecipe }) {
  if (subrecipe) {
    return <Link href={`/recipes/${subrecipe.slug}`}>{name}</Link>;
  }
  return name;
}
Enter fullscreen mode Exit fullscreen mode

It might be a personal pet peeve, but I dislike when the return statements are on different levels, making them harder to compare. Let’s add an else statement to fix this:

function RecipeName({ name, subrecipe }) {
  if (subrecipe) {
    return <Link href={`/recipes/${subrecipe.slug}`}>{name}</Link>;
  } else {
    return name;
  }
}
Enter fullscreen mode Exit fullscreen mode

Now, both return values are at the same indentation level, making them easier to compare. This pattern works when none of the condition branches are handling errors, in which case an early return would be a better approach.

Info: We talk about early returns in the Avoid conditions chapter.

Here’s another example:

<Button
  onPress={Platform.OS !== 'web' ? onOpenViewConfirmation : undefined}
  link={Platform.OS === 'web' ? previewLink : undefined}
  target="_empty"
>
  Continue
</Button>
Enter fullscreen mode Exit fullscreen mode

In this example, we have a button that behaves like a link in the browser and shows a confirmation modal in an app. The reversed condition for the onPress prop makes this logic hard to see.

Let’s make both conditions positive:

<Button
  onPress={Platform.OS === 'web' ? undefined : onOpenViewConfirmation}
  link={Platform.OS === 'web' ? previewLink : undefined}
  target="_empty"
>
  Continue
</Button>
Enter fullscreen mode Exit fullscreen mode

Now, it’s clear that we either set onPress or link props depending on the platform.

We can stop here or take it a step further, depending on the number of Platform.OS === 'web' conditions in the component or how many props we need to set conditionally

We can extract the conditional props into a separate variable:

const buttonProps =
  Platform.OS === 'web'
    ? {
        link: previewLink,
        target: '_empty'
      }
    : {
        onPress: onOpenViewConfirmation
      };
Enter fullscreen mode Exit fullscreen mode

Then, use it instead of hardcoding the entire condition every time:

<Button {...buttonProps}>Continue</Button>
Enter fullscreen mode Exit fullscreen mode

I also moved the target prop to the web branch because it’s not used by the app anyway.


When I was in my twenties, remembering things wasn’t much of a problem for me. I could recall books I’d read and all the functions in a project I was working on. Now that I’m in my forties, that’s no longer the case. I now value simple code that doesn’t use any tricks; I value search engines, quick access to the documentation, and tooling that help me to reason about the code and navigate the project without keeping everything in my head.

We shouldn’t write code for our present selves but for who we’ll be a few years from now. Thinking is hard, and programming demands a lot of it, even without having to decipher tricky or unclear code.

Start thinking about:

  • When you feel smart and write some short, clever code, think if there’s a simpler, more readable way to write it.
  • Whether a condition that makes code slightly different is truly necessary.
  • Whether a shortcut makes the code shorter but still readable, or just shorter.

If you have any feedback, mastodon me, tweet me, open an issue on GitHub, or email me at artem@sapegin.ru. Get your copy.

Top comments (17)

Collapse
 
jonrandy profile image
Jon Randy 🎖️ • Edited

Another obscure use of the bitwise NOT operator is to discard the fractional portion of a number. Use Math.floor() instead:

Be VERY careful here. Math.floor and ~~ are not the same thing, and are not interchangeable. Math.trunc and ~~ ARE pretty much the same though.

Collapse
 
josefjelinek profile image
Josef Jelinek

Math.trunc() works on 64 bit floats (double), while ~~ works on 32 bit integers (JS converts it before applying the bitwise operators), so

Math.trunc(5000000000) => 5000000000
Enter fullscreen mode Exit fullscreen mode

while

~~5000000000 => 705032704
Enter fullscreen mode Exit fullscreen mode
Collapse
 
jonrandy profile image
Info Comment hidden by post author - thread only accessible via permalink
Jon Randy 🎖️ • Edited

👍 Yup! That's why I said 'pretty much' the same (didn't have time to write out the full explanation - thanks for doing that).

It's also worth noting that if you're doing a large amount of processing, and the precision (or lack thereof) of using ~~ is acceptable... then you should probably use it as it can be (environment dependent) much faster than Math.trunc... on Chrome it is about 50% faster.

Sometimes performance trumps 'cleanliness' in requirements.

 
josefjelinek profile image
Josef Jelinek

Sure. However, I think in that case, you will likely use some more performant way like in32 array buffer to store results, since it would likely matter only when you do a lot of processing of this particular type, otherwise both operations would disappear in the noise...

You should try adding this to your test cases for a big surprise:

const out = new Int32Array(data.length)
for (let i = 0; i < data.length; i++) {
  out[i] = data[i]
}
Enter fullscreen mode Exit fullscreen mode
Collapse
 
sapegin profile image
Artem Sapegin

More reasons to avoid ~~!

Collapse
 
jonrandy profile image
Info Comment hidden by post author - thread only accessible via permalink
Jon Randy 🎖️ • Edited

We should not just 'blanket ban' it without understanding it, and knowing that it could be advantageous. Depending on what you are doing, ~~ may be a superior choice - it is MUCH faster (up to 50% faster on Chrome). I know people will say that premature optimisation is evil, or that we shouldn't favour performance over readability... but that is not always true and sometimes speed IS essential.

Dogmatically preaching like this decreases understanding of the language and makes for less capable developers.

 
sapegin profile image
Artem Sapegin

Dogmatically preaching like this

Lol

Collapse
 
sapegin profile image
Artem Sapegin

I did change it though, so thanks for pointing this out:

github.com/sapegin/washingcode-boo...

Collapse
 
open768 profile image
sunil

I do a lot of reverse engineering for work when the original developers have left and no-one truly knows how the code does what it does. I'm brought in to give new developers something to start with, to understand where to start to look in the code.

A couple of things i'm a fan of is

  • //------------------------------------------------------------------------------ visual breaks in the code, with comment lines that have no purpose other than separating a method or class from the next one //------------------------------------------------------------------------------
  • variable unpacking. yes the language allows method1()->method2()->method3(), but it makes it vary hard to read what the intent of the programmer was. its more readable to break the steps into broken out statements. chained methods are not necessarily faster, as the compiler will do the better optimisations than a typical coder can.
Collapse
 
pengeszikra profile image
Peter Vivo • Edited

Imho I using that paralell code as arrow function, which has only one return.
In my taste is ; put on same column as ? and : because in later much easier change that two lines and don't need to be worry about ; after name

const RecipeName = ({ name, subrecipe }) => 
  (subrecipe) 
    ? <Link href={`/recipes/${subrecipe.slug}`}>{name}</Link>
    : name
    ;
Enter fullscreen mode Exit fullscreen mode
Collapse
 
sapegin profile image
Artem Sapegin

This could be a good topic to discuss some ten years ago, and I do agree it's very readable. Now with Prettier it doesn't make any difference though.

(There's a chapter on code style in the book with similar examples.)

Collapse
 
jonrandy profile image
Jon Randy 🎖️

Example 4 is wrong. The two snippets of code do 2 different things

Collapse
 
sapegin profile image
Artem Sapegin

Claiming something is wrong without explaining why isn't very helpful.

Collapse
 
jonrandy profile image
Jon Randy 🎖️ • Edited

You need || not &&. The second example is checking that both arrays are not empty. The first just checks that at least one has content.

My preference here would be:

if (dogs.length || cats.length) {
...
Enter fullscreen mode Exit fullscreen mode
Thread Thread
 
sapegin profile image
Artem Sapegin

You're right! And yet — another reason to avoid clever code ;-)

Thread Thread
 
jonrandy profile image
Info Comment hidden by post author - thread only accessible via permalink
Jon Randy 🎖️

But the problem was in your 'clean' version. You had no problem describing what the 'non-clean' version did

Collapse
 
jonrandy profile image
Info Comment hidden by post author - thread only accessible via permalink
Jon Randy 🎖️

A balance needs to be struck here though... Avoiding 'clever' code can actually have a detrimental effect on the quality of developers over time (and also on the efficiency of the code):

Some comments may only be visible to logged-in visitors. Sign in to view all comments. Some comments have been hidden by the post's author - find out more