最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - svelte: how can I declare two cyclically reactive variables? - Stack Overflow

programmeradmin1浏览0评论

I have two variables a and b which add up to 100. How do I set up a reactive declaration such that when a changes b changes to be 100 - a and vice versa? When I try something like

let total = 100;
$: a = total - b;
$: b = total - a;

I get a 'Cyclical dependency detected' error. Is there any way to get this done?

I have two variables a and b which add up to 100. How do I set up a reactive declaration such that when a changes b changes to be 100 - a and vice versa? When I try something like

let total = 100;
$: a = total - b;
$: b = total - a;

I get a 'Cyclical dependency detected' error. Is there any way to get this done?

Share Improve this question edited Mar 25, 2022 at 21:00 Austin Walela asked Mar 25, 2022 at 20:51 Austin WalelaAustin Walela 3351 gold badge5 silver badges19 bronze badges 0
Add a ment  | 

3 Answers 3

Reset to default 12

The problem es from the fact that Svelte wants to sort the reactive blocks in the order they depend upon each other: it wants to pute those that are depended upon but have no dependencies first, to avoid unneeded putation... or getting trapped in a loop.

Furthermore, Svelte considers a dependency of a reactive expression any reactive variable that appear in it.

And so, the actual source of your error is that both variables a and b appears in both reactive expressions.

The solution is to remove from the reactive expressions the unneeded variable, that is the one that is assigned to. This can be done by moving the assignment to a function outside of the reactive block. Unfortunately, it's more verbose...

<script>
    let total = 100;
    let a = 0
    let b = 0
    
    const setA = () => {
        // the assignment to a (or b in the other function) are still
        // reactive, of course, but Svelte won't propagate changes to
        // variable that are assigned their current value, so this
        // will break the loop
        a = total - b
    }
    
    const setB = () => {
        b = total - a
    }
    
    $: setA(total - b);
    $: setB(total - a);
</script>

<pre>
    a: {a}
    b: {b}
</pre>

<label>
    a <input type="number" bind:value={a} />
</label>

<label>
    b <input type="number" bind:value={b} />
</label>

Edit

As remarked by @avf in the ments, the above code is misleading and not so obvious for the reader of the code. I wrote it like this at the time because I was focused on demonstrating the principles of Svelte reactivity.

In a real world context, I would rather remend any of the following two forms.

Whenever possible, favour obviousness:

// this function is just normal
const setA = (value) => {
  a = value
}

// this reactive expression makes it obvious and straightforward that 
// its dependencies are total and b, and only those
$: setA(total - b)

When obviousness is not an option (for example because the content of the function would be more side effect than mere assignment), I'd rely on the following form that is very idiomatic to Svelte. Unfortunately, it's also very obscure to people who are not familiar to Svelte usually... But I guess that just belongs to those things you've got to learn to be really proficient with the framework.

const reputeEverythingOrWhatever = () => {
  ...
}

// in Svelte lingo, this is broadly understood as "whenever a, or b, or
// total changes, then repute everything (or whatever)"
$: a, b, total, reputeEverythingOrWhatever()

This syntax is fine IMO because it is generally familiar and well understood by Svelte developers.

Be careful however with what you put in the repute function. If it changes the values of the dependencies of the reactive block, the code will pile fine this time, and Svelte would even break update loops at runtime to avoid freezing the JS thread... But you might still end up with code that will be very hard to prehend or predict.

Unless you need the values to update from programmatic changes, i would drop the reactive statements entirely and work with events instead: When one variable is adjusted by the user, update the other one.

<script>
  const total = 100;
  let a = total;
  let b = 0;
</script>

a: {a} <br>
b: {b}

<label>
  a <input type="number" bind:value={a}
           on:input={() => b = total - a} />
</label>

<label>
  b <input type="number" bind:value={b}
           on:input={() => a = total - b} />
</label>

That does not work; Svelte does not allow this and this cycle cannot be resolved either. If you want a variable to be editable, you cannot declare it as reactive ($: x = ...). You can either reactively set regular variables with reactive statements/blocks or use events instead (see other answers).

The following is an explanation why this could not be resolved logically either.

Treating this like two equations with two unknowns you get this useless simplification:

a = total - b
b = total - a
b = total - (total - b)
b = total - total + b
b = b

You have to fix at least one of the values, otherwise all possible values are valid here.


You can also first normalize the equations:

a = total - b => a + b = total => a + b = total
b = total - a => b + a = total => a + b = total

As you can see, they are the same, so you actually have two unknowns and only one equation, so this is underspecified.

(Note that even if this would yield a valid solution, Svelte cannot solve systems of linear equations for you.)

发布评论

评论列表(0)

  1. 暂无评论