Demo: http://undesirable-bucket.surge.sh

git clone https://github.com/drcmda/scheduler-test
cd scheduler-test
yarn
yarn start

⚠️

It depends on react@experimental, react-reconciler@experimental and scheduler@experimental.

Explanation

This project creates a highly taxing environment for concurrent React.

It simulates heavy load by creating hundreds of THREE.TextGeometry instances (510 to be exact). This class, like many others in Threejs, is expensive and takes a while to construct. If all 510 instances are created the same time it will cause approximately 1.5 seconds of pure jank (Apple M1), the tab would normally freeze. It runs in an interval and will execute every 2 seconds.

This is the framework-independent vanilla stress-test:

async function test() {
  const chars = `!"§$%&/()=?*#<>-_.:,;+0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz`
  const font = await new Promise((res) => new THREE.FontLoader().load("https://raw.githubusercontent.com/drcmda/scheduler-test/master/public/Inter%20UI_Bold.json", res))
  console.time("test")
  for (let i = 0; i < 510; i++) {
    new THREE.TextGeometry(chars[Math.floor(Math.random() * chars.length)], {
      font,
      size: 1,
      height: 0.5,
      curveSegments: 80,
      bevelEnabled: false,
    })
  }
  console.timeEnd("test")
}

test()

// To really drive it home you'd have to repeat it every two seconds ...
// setInterval(test, 2000)

Try it out yourself: https://codesandbox.io/s/distracted-hypatia-6kqxh?file=/src/index.js (warning, it may freeze your tab).

Expectation

Reacts has to schedule that load so that more or less 60 frames/second can always be maintained. If it actually hits 60 fps still depends on your machine and GPU, these are a lot of drawcalls after all, but at least the frame rate must be stable.

There are two modes:

  • Distributed: the app will randomly spread the creation of all instances over a second, by the end of which all instances must be created. This simulates ongoing stress
  • At-once (distributed is off): all instances will be created at once, which is the worst case scenario.

Every time a block is being regenerated it flashes up red.

Observations and stats (Apple M1)

Plain Threejs (or legacy React) is simulated when concurrent is off, otherwise React will run in concurrent mode.

Distributed At-once
Threejs ~20fps ~5fps
React ~60fps ~60fps

How?

React can do this because of its new scheduler, which supports concurrency. Think of how a virtual list schedules its items, no matter if you give it 10 or 10.000.000, it will render only as much as the screen can take. But React does this at the base level, every operation is weighed. If operations would start to bite into the framerate React must balance them.

Watch this talk by Dan Abramov if you want to know how it works: https://www.youtube.com/watch?v=nLF0n9SACd4

Can this be done without React?

Probably not. To get anything even remotely similar in user-land would be an endevour. This is not just simple requestIdleFrame and defer, think of the repercussions: mount/unmount race conditions, async ops, everything must still function in scale.