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

[Performance] Slice contexts (createSliceWithContext) #2202

Closed
gentlee opened this issue Sep 26, 2024 · 7 comments
Closed

[Performance] Slice contexts (createSliceWithContext) #2202

gentlee opened this issue Sep 26, 2024 · 7 comments

Comments

@gentlee
Copy link

gentlee commented Sep 26, 2024

What is the new or updated feature that you are suggesting?

Problem:

One of the slowest parts of redux is that there can be a lot of subscribers to the store with their custom state comparers. If we have 1000 subscribers and 100 actions are triggered per second (like on startup of the chatting app), 100 000 comparisons are made per second.

One possible solution is to use multiple stores, but this can lead to various issues.

Suggestion:

The same performance of having multiple stores can be achieved with single store. There is already an ability to make custom contexts and hooks, but currently it makes sense only for multiple stores.

UPDATED: Ability to provide the same store but with slice's context and hooks should be added with the new createSliceWithContext function:

export const {
  // new prop for slice selectors - ones that select from slice state, not the root
  // basically the same provided while creating the slice, nothing new here - just pass them here
  sliceSelectors: { 
    ...
  },
  // usual selectors can be kept as well
  selectors: { ... }
  // new props for context and provider
  context: AccountSliceContext,
  provider: AccountSliceProvider,
  hooks: {
    // new hook that connects to the new context with the slice state
    useSelector: useAccountSelector,
  },
  // other returned values are the same
  ...
} = createSliceWithContext({
  // everything is the same as in usual createSlice 
  name: 'account',
  selectors: {
    ... 
  },
  ...
})

const App = () => {
  return (
    <Provider store={store} />
      <AccountSliceProvider store={store}>
        <RootModule />
      </AccountSliceProvider>
    </Provider>
  );
}

const RootModule = () => {
  // here we got the state of the account state, not the whole redux state
  // this selector should not be executed and compared on redux state updates when account state does not change
  const accountState = useAccountSelector(accountState => accountState, customEqualityFn) 

  return null
}

Please also check the initial suggestion as it provides more generic solution, and can be implemented as well / instead:

OLD Ability to provide the same store but with domain selector should be added:
const store = createStore(reducer);

const SomeDomainContext = React.createContext(null);
const someDomiainSelector = (state) => state.someDomain

// custom hook for a specific domain
export const useSomeDomainSelector = createSelectorHook(MyContext, someDomainSelector)

const App = () => {
  return (
    <Provider store={store} />
      <Provider store={store} selector={someDomainSelector} context={SomeDomainContext}>
        <RootModule />
      </Provider>
    </Provider>
  );
}

const RootModule = () => {
  // here we got the state of the selected domain, not the whole redux state
  // this selector should not be executed and compared on redux state updates when domain does not change
  const someDomainState = useSomeDomainSelector(domainState => domainState, customEqualityFn) 

return null
}

Why should this feature be included?

This will make using multiple stores totally obsolete, but will fix performance issues caused by single store.

What docs changes are needed to explain this?

https://react-redux.js.org/using-react-redux/accessing-store

Add recommended practise here:
https://redux.js.org/style-guide/

@markerikson
Copy link
Contributor

Agreed that the O(n) behavior of Redux's subscription model is both a strength (simplicity) and a weakness (perf at very large scales).

That said, if I'm understanding this proposal right, the approach here isn't viable. There's no guarantee that a given subtree will only ever access a specific slice of state. The whole point of useSelector is that any component can read from any part of the state at any time.

@markerikson markerikson closed this as not planned Won't fix, can't repro, duplicate, stale Sep 26, 2024
@gentlee
Copy link
Author

gentlee commented Sep 26, 2024

@markerikson you can see that someDomiainSelector is passed to createSelectorHook too, and that makes the hook provide only domain state, not the whole.

So yes, it will access only a specific slice.

And yes, this optimization makes sense only for huge apps.

@markerikson
Copy link
Contributor

markerikson commented Sep 26, 2024

Yeah, that's not something we plan to implement ourselves.

(I do have a general interest in ideas to optimize React-Redux perf, but it's not something I have time to look into atm.)

@gentlee gentlee changed the title [Performance] Domain contexts & selectors [Performance] Slice contexts Sep 26, 2024
@gentlee gentlee changed the title [Performance] Slice contexts [Performance] Slice contexts (createSliceWithContext) Sep 26, 2024
@gentlee
Copy link
Author

gentlee commented Sep 26, 2024

@markerikson OK, I updated header a bit with idea to create a createSliceWithContext, for future if someone decides to do that.

@gentlee
Copy link
Author

gentlee commented Sep 27, 2024

Other possible ways to reach that:

  1. combineStores into single same as combineReducers / combineSlices - that way we also separate action handling by reducers for multiple stores if dispatched not by the main store, but by the child store. Which is a nice feature to have.
  2. Some wrapper (e.g. createDomain(reducer)) around reducer (including combined reducers) that returns context, hooks (useSelector, useDispatch can also be returned to separate actions only for that domain, if it is possible) and mb even selectors (if coming from combined slices) for the new context.

@phryneas
Copy link
Member

Just to throw the idea out - with something like proxy-memoize, there could be a new version of useSelector that shared a single store subscription and then would trigger subscribers much more granularly, but transparently to the user.

@markerikson
Copy link
Contributor

markerikson commented Sep 27, 2024

Yeah, that's roughly similar to what I was playing around with in this PR:

but A) I haven't ever had time to go back and play with that idea again, B) I'm pretty sure there's a ton of edge cases this would have issues with, and C) it requires an up-front fixed set of work (reconciling the immutably updated state tree into the proxy wrapper) and the cost of reading from proxies (which I recently found out can be relatively expensive), and it's very questionable if the amount of time saved from skipping subscribers and selectors would actually be a net improvement in perf.

I can say that apparently Slack did do something similar to this "domains" proposal in terms of watching for top-level slice changes, but it sounds like they did it by wrapping the store subscription process itself and providing methods for consumers to subscribe to just those keys, thus providing some fan-out behavior. (Also, evidence that this can be done in userland, and thus not something I would want to build in to the lib itself any time soon.)

In fact, I actually just went back and looked at the list of Redux addons I kept updated between 2016 and 2018, and I see several plausible-sounding "only subscribe to a specific piece of the state" libs here:

Alternately, @gentlee , you might want to look at https://www.npmjs.com/package/redux-subspace . It's deprecated, but I think the API is actually pretty similar to what you're asking for, and the README lists a few other alternative libraries.

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

No branches or pull requests

3 participants