Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: Faster method of building dependencies array #824

Merged
merged 1 commit into from
Aug 30, 2021

Conversation

paddyobrien
Copy link
Contributor

Motivation

Fixes #797

Summary

I'm pretty surprised that this has any impact, I don't have a good rationale for why but it appears to be dramatically faster to update the array in place.

Using the demo repo linked in the above issue without this change:

$ webpack
asset app.js 961 KiB [compared for emit] (name: app) 1 related asset
asset styles.css 8.35 KiB [compared for emit] (name: app) 1 related asset
Entrypoint app 969 KiB (1.11 MiB) = styles.css 8.35 KiB app.js 961 KiB 2 auxiliary assets
orphan modules 390 bytes [orphan] 3 modules
runtime modules 670 bytes 3 modules
modules by path ./node_modules/ 947 KiB
  modules by path ./node_modules/scheduler/ 32.4 KiB 4 modules
  modules by path ./node_modules/react/ 59.4 KiB 2 modules
  modules by path ./node_modules/react-dom/ 840 KiB 2 modules
  modules by path ./node_modules/@linaria/ 4.34 KiB 2 modules
  modules by path ./node_modules/@emotion/ 4.41 KiB 2 modules
  modules by path ./node_modules/prop-types/ 4 KiB 2 modules
  ./node_modules/object-assign/index.js 2.06 KiB [built] [code generated]
modules by path ./.linaria-cache/src/*.css 50 bytes (javascript) 8.31 KiB (css/mini-extract)
  ./.linaria-cache/src/App.linaria.css 50 bytes [built] [code generated]
  css ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[1].use[1]!./.linaria-cache/src/App.linaria.css 8.31 KiB [built] [code generated]
./src/App.js 4.68 KiB [built] [code generated]
webpack 5.42.0 compiled successfully in 23904 ms

And with this change:

$ webpack
asset app.js 961 KiB [emitted] (name: app) 1 related asset
asset styles.css 8.35 KiB [compared for emit] (name: app) 1 related asset
Entrypoint app 969 KiB (1.11 MiB) = styles.css 8.35 KiB app.js 961 KiB 2 auxiliary assets
orphan modules 390 bytes [orphan] 3 modules
runtime modules 670 bytes 3 modules
modules by path ./node_modules/ 947 KiB
  modules by path ./node_modules/scheduler/ 32.4 KiB 4 modules
  modules by path ./node_modules/react/ 59.4 KiB 2 modules
  modules by path ./node_modules/react-dom/ 840 KiB 2 modules
  modules by path ./node_modules/@linaria/ 4.34 KiB 2 modules
  modules by path ./node_modules/@emotion/ 4.41 KiB 2 modules
  modules by path ./node_modules/prop-types/ 4 KiB 2 modules
  ./node_modules/object-assign/index.js 2.06 KiB [built] [code generated]
modules by path ./.linaria-cache/src/*.css 50 bytes (javascript) 8.31 KiB (css/mini-extract)
  ./.linaria-cache/src/App.linaria.css 50 bytes [built] [code generated]
  css ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[1].use[1]!./.linaria-cache/src/App.linaria.css 8.31 KiB [built] [code generated]
./src/App.js 4.68 KiB [built] [code generated]
webpack 5.42.0 compiled successfully in 1874 ms

@callstack-bot
Copy link

Hey @paddyobrien, thank you for your pull request 🤗.
The coverage report for this branch can be viewed here.

@Anber
Copy link
Collaborator

Anber commented Aug 30, 2021

Hi @paddyobrien
This is an incredible catch! I can't explain why concat+reduce has such an incredible complexity when it runs in webpack5 context. Probably, webpack 5 somewhat tries to manage memory allocations or garbage-collections which reduce+concat produces quite a lot.
Anyway, thank you so much!

@Anber Anber merged commit 2463881 into callstack:master Aug 30, 2021
@paddyobrien
Copy link
Contributor Author

Thanks for merging @Anber I was really surprised this was so much faster but a process of elimination led me to it and I was left with no other conclusion than to change these lines.

Would it be possible to cut another beta release so we can check with our production app if there are any other performance regressions?

@Anber
Copy link
Collaborator

Anber commented Aug 31, 2021

@paddyobrien I'll publish the next beta later today.

@eoin
Copy link

eoin commented Nov 18, 2021

@paddyobrien and I weren't very satisfied with our understanding of the root cause here so we spent some time digging 😄
It turned out to be pretty interesting. Here's a writeup.

@rcstr
Copy link

rcstr commented Nov 19, 2021

I enjoyed a lot the write-up @eoin. thanks a lot!

@davemilter
Copy link

davemilter commented Nov 25, 2021

@paddyobrien

I'm pretty surprised that this has any impact, I don't have a good rationale for why but it appears to be dramatically faster to update the array in place.

But this code on each iteration create new Array and copy result to it:

return nodes.reduce(
  (acc, node) => acc.concat(Array.from(this.dependencies.get(node) || [])),

with length from one 1 to n, where n length of nodes.
So this is n * (n + 1) operation, in other words complexity of initial algorithm is O(n^2).
So obviously the new algorithm with complexity O(n) is n times faster.

So it should be pretty surprised if there are dramatically impact of such changes.
And gives thanks to v8 who some times can optimize such slow code.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Slow build on Webpack 5
6 participants