Skip to content
Advertisement

Why does a derived Svelte store have different behaviour when using `$` vs `subscribe`

I have a data model that I can’t change in this project. I’m trying to strip down and simplify the sample code below, so hopefully this still makes sense with what I’m trying to re-produce.

Let’s say I have two stores. One store holds “containers” and the other store holds “items” – each store is independently used throughout the app for various purposes. The Container only holds Item IDs by default. Occasionally, I want to de-normalize the data for use on some pages. I use a derived store and denormalize each of the container objects, turning them into DenormalizedContainers.

Let’s say I have a use case where I want to create a new Item object, and then add it to a given Container. As far as I can tell, that will cause the derived store to update twice (once for the change to items and again for a change to containers), and thus anything that uses that derived store will update twice (depending on how it’s called).

Why would the behaviour be different from subscribe to $?

Also, while this isn’t a huge problem for me, I’m just curious if there is any native Svelte-y way to workaround this without changing the data model (while still being able to use the subscribe API?

JavaScript

Results:

Upon clicking each button once, here are the logs:

Page load:

JavaScript

Add container:

JavaScript

Add item:

JavaScript

Add both:

JavaScript

The updates handled via $ are what I would like all the time, but for some reason the manual subscribe API shows two updates. I have a couple locations outside of .svelte files, where I would need to use the subscribe API.

References:

Similar to this, but doesn’t explain why the $ behaves differently to the subscribe.

Svelte Derived Store atomic / debounced updates

Possibly the underlying problem?

https://github.com/sveltejs/svelte/issues/6730

Possibly explained in 3.1/3.2 here. Micro-tasks bundle up reactive statements and apply them at the end of the tick. Whereas subscribe handles updates in real-time. Note: Wrapping my subscribe code in a tick just makes it run twice with the same data.

https://dev.to/isaachagoel/svelte-reactivity-gotchas-solutions-if-you-re-using-svelte-in-production-you-should-read-this-3oj3

Advertisement

Answer

If you look at the compiled output you can see that the reactive statements are triggered on update, which should only happen once per event loop.

JavaScript

You can create your own method of bundling updates in an event loop. E.g. using micro tasks:

JavaScript

REPL example

There is no way to cancel micro tasks again, so the old ones are invalidated via a key. Other ways of ensuring that only the last task is executed might be more elegant.

With setTimeout this can be done as well, and with that the previous one can be cancelled, but the minimum delay will be larger:

JavaScript
User contributions licensed under: CC BY-SA
9 People found this is helpful
Advertisement