How to efficiently batch DOM updates to reduce browser reflow?

Content verified by Anycode AI
July 21, 2024
A big part of what makes modern web development work is ensuring user experiences are as seamless as can be. Critical among these is the minimization of browser reflows—a process by which a browser recalculates the position and geometry for every element on a web page as its DOM changes. If continuous, it creates high performance degradation, in turn leading to janky animations and slow interactions. This guide solves this problem by offering strategies to batch DOM updates efficiently for fewer occurrences of such reflows. It will allow developers to enhance their web application performance and improve the satisfaction of the end user by strchrizing and implementing mechanisms like Document Fragments, \`requestAnimationFrame\`, debouncing, and optimal CSS practices.

To really get a grip on optimizing those troublesome DOM updates and reducing the browser's dreaded reflow and repaint processes, let's dive into how our dear browser handles these changes to the Document Object Model (DOM). Every time you tweak the DOM, it could lead to a reflow (layout recalculations) or a repaint (refreshing the visuals). These tasks aren't cheap in terms of performance, so the fewer we have, the better!

When a webpage layout needs recalculating, that's called reflow. This happens whenever changes affect an element’s position or size, stirring the browser to recalculate its rendering. Repaint is slightly different; it involves updating the visuals without affecting layout geometry – think of changing a background color. Since both operations are costly, cutting down their frequency can make our web pages fly rather than crawl.

Minimize Layout Thrashing

Layout thrashing is like a traffic jam; it happens when reading and writing to properties that force layout calculations are mixed. To end this bottleneck, aim for grouping such operations together.

const el = document.getElementById('example');

// Layout Thrashing (inefficient way)
for(let i = 0; i < 10; i++) {
  el.style.width = (el.offsetWidth * 1.1) + 'px'; // Causes reflow
  el.style.height = (el.offsetHeight * 1.1) + 'px'; // Causes reflow
}

// Optimized way
const width = el.offsetWidth; // Single read for width
const height = el.offsetHeight; // Single read for height

el.style.width = (width * 1.1) + 'px';
el.style.height = (height * 1.1) + 'px';

Batch DOM Changes

Act like an efficient chef in a kitchen—batch your DOM changes. This involves making several updates in one go, avoiding those nasty intermediate reflows. Here’s a neat trick with DocumentFragment.

const container = document.getElementById('container');
const fragment = document.createDocumentFragment();

for (let i = 0; i < 100; i++) {
  const newElement = document.createElement('div');
  newElement.textContent = 'Item ' + i;
  fragment.appendChild(newElement);
}

container.appendChild(fragment); // Single reflow moment

Use requestAnimationFrame

Whenever you’re grouping updates, requestAnimationFrame becomes your best buddy. This function lets the browser handle changes during its next paint cycle, streamlining the process.

const updateDOM = () => {
  const el = document.getElementById('example');
  el.style.width = '500px';
  el.style.height = '500px';
  // Further DOM updates
};

window.requestAnimationFrame(updateDOM);

Avoid Inline Styles and Use CSS Classes

Switching up inline styles frequently can be a performance killer. Adding or removing CSS classes is a much smoother move.

// Inefficient approach
el.style.left = '100px';
el.style.top = '100px';

// A better way
el.classList.add('new-position');

CSS:

.new-position {
  left: 100px;
  top: 100px;
}

Minimize or Delay Complex Layouts

If you've got heavy layout tasks, consider hiding elements while making changes, then reveal them when you’re done tweaking.

const element = document.getElementById('content');
element.style.display = 'none'; // Temporarily hide

// Perform complex modifications
element.innerHTML = largeUpdatedContent;

element.style.display = 'block'; // Reveal after changes

Use Efficient Selectors and Caching

Accessing and modifying the DOM efficiently can spare you from needless reflows.

// Quite inefficient 
for (let i = 0; i < 100; i++) {
  document.getElementById('item-' + i).style.color = 'blue';
}

// Much better
const container = document.getElementById('container');
const items = container.querySelectorAll('.item');

items.forEach(item => {
  item.style.color = 'blue';
});

Out-of-Document Updates

Making updates away from the live DOM tree, then appending it back later, can also save time.

const parent = document.createElement('div');

for (let i = 0; i < 100; i++) {
  const child = document.createElement('div');
  child.textContent = 'Item ' + i;
  parent.appendChild(child);
}

document.body.appendChild(parent); // Only one reflow

Use MutationObserver to Manage Batch Updates

MutationObserver lets you keep an eye on changes to the DOM and batch them as needed.

const observer = new MutationObserver((mutationsList) => {
  requestAnimationFrame(() => {
    mutationsList.forEach(mutation => {
      // Handle mutations here
    });
  });
});

const config = { childList: true, attributes: true, subtree: true };
const targetNode = document.getElementById('container');

observer.observe(targetNode, config);

// Make some DOM updates 
targetNode.textContent = 'Some update';
targetNode.setAttribute('style', 'color: red;');

// You can later disconnect the observer
observer.disconnect();

Planning those DOM interactions with precision, reading and writing DOM properties wisely, utilizing intermediate structures like DocumentFragment, using requestAnimationFrame for smooth updates, and preferring classes over inline styles for CSS modifications—these techniques can significantly contribute to a smoother, faster, more efficient web experience.

Have any questions?
Our CEO and CTO are happy to
answer them personally.
Get Beta Access
Anubis Watal
CTO at Anycode
Alex Hudym
CEO at Anycode