Reaktivita podrobně
Jednou z nejvýraznějších vlastností Vue je nenápadný systém reaktivity. Stav komponenty se skládá z reaktivních JavaScript objektů. Když je upravíte, zobrazení se aktualizuje. To dělá správu stavu jednodušší a intuitivní. Je však také důležité pochopit, jak to funguje, abyste se vyhnuli běžným problémům. Na této stránce se budeme zabývat některými podrobnostmi reaktivního systému Vue na nižší úrovni.
Co je reaktivita?
Tento termín se v programování často používá, ale co tím lidé myslí? Reaktivita je programovací paradigma, které nám umožňuje přizpůsobit se změnám deklarativním způsobem. Klasickým příkladem, který lidé obvykle ukazují, protože je velmi názorný, je tabulkový procesor Excel:
A | B | C | |
---|---|---|---|
0 | 1 | ||
1 | 2 | ||
2 | 3 |
Zde je buňka A2 definována pomocí vzorce = A0 + A1
(kliknutím na A2 si můžete zobrazit nebo upravit vzorec), takže tabulkový procesor nám dává výsledek 3. Žádné překvapení. Když však aktualizujete A0 nebo A1, všimnete si, že A2 se také automaticky aktualizuje.
JavaScript obvykle takto nefunguje. Kdybychom chtěli napsat něco srovnatelného v JavaScriptu:
js
let A0 = 1
let A1 = 2
let A2 = A0 + A1
console.log(A2) // 3
A0 = 2
console.log(A2) // Stále 3
Když změníme A0
, A2
se automaticky nezmění.
Jak bychom to tedy v JavaScriptu udělali? Abychom znovu spustili kód, který aktualizuje A2
, zabalme ho nejprve do funkce:
js
let A2
function update() {
A2 = A0 + A1
}
Potom musíme definovat několik pojmů:
Funkce
update()
produkuje vedlejší efekt (side effect), nebo zkráceně efekt, protože mění stav programu.A0
aA1
jsou považovány za závislosti (dependencies) efektu, protože jejich hodnoty se používají k provedení efektu. Efekt je nazýván odběratelem (subscriber) svých závislostí.
Co potřebujeme, je magická funkce, která může vyvolat update()
(efekt) pokaždé, když se změní A0
nebo A1
(závislosti):
js
whenDepsChange(update)
Tato funkce whenDepsChange()
má následující úkoly:
Sledovat, když je proměnná čtena. Například při vyhodnocování výrazu
A0 + A1
jsou čteny obě proměnnéA0
aA1
.Pokud je proměnná čtena, když je právě spuštěný efekt, stane se tento efekt odběratelem této proměnné. Například protože jsou proměnné
A0
aA1
čteny při vykonáváníupdate()
, po prvním volání seupdate()
stane odběratelem obou proměnnýchA0
aA1
.Detekovat, když se proměnná změní. Například když je proměnné
A0
přiřazena nová hodnota, upozornit všechny efekty odběratelů, aby se znovu spustily.
Jak funguje reaktivita ve Vue
Ve skutečnosti nemůžeme čtení a zápis lokálních proměnných sledovat jako v příkladu. V čistém JavaScriptu pro to prostě neexistuje mechanismus. Co ale můžeme udělat, je zachytit čtení a zápis vlastností objektů.
Existují dva způsoby, jak zachytit přístup k vlastnostem v JavaScriptu: getter / setter funkce a Proxies. Vue 2 používalo výhradně getters / setters kvůli omezením podpory prohlížečů. Ve Vue 3 se pro reaktivní objekty používají proxies a pro refs se používají getters / setters. Zde je pseudokód, který ilustruje, jak fungují:
js
function reactive(obj) {
return new Proxy(obj, {
get(target, key) {
track(target, key)
return target[key]
},
set(target, key, value) {
target[key] = value
trigger(target, key)
}
})
}
function ref(value) {
const refObject = {
get value() {
track(refObject, 'value')
return value
},
set value(newValue) {
value = newValue
trigger(refObject, 'value')
}
}
return refObject
}
TIP
Ukázky kódu zde a níže mají za úkol vysvětlit základní koncepty co nejjednodušším způsobem, takže je mnoho detailů vynecháno a okrajové případy jsou ignorovány.
Toto vysvětluje několik omezení reaktivních objektů, o kterých jsme mluvili v kapitole základů:
Když přiřadíte nebo destruujete vlastnost reaktivního objektu do místní proměnné, přístup nebo přiřazení k této proměnné není reaktivní, protože již nevyvolává get / set proxy traps na zdrojovém objektu. Toto „odpojení“ však ovlivňuje pouze vazbu proměnné. Pokud proměnná odkazuje na neprimitivní hodnotu, jako je objekt, změna objektu reaktivní pořád bude.
Vrácená proxy z
reactive()
, i když se chová stejně jako originál, má odlišnou identitu, pokud ji porovnáme s původním objektem pomocí operátoru===
.
Uvnitř track()
kontrolujeme, zda je právě spuštěný efekt. Pokud ano, vyhledáme efekty odběratelů (uložené v Set kolekci) pro sledovanou vlastnost a přidáme efekt do kolekce:
js
// Toto bude nastaveno před spuštěním efektu.
// Vrátíme se k tomu později.
let activeEffect
function track(target, key) {
if (activeEffect) {
const effects = getSubscribersForProperty(target, key)
effects.add(activeEffect)
}
}
Odběry efektů jsou uloženy ve globální datové struktuře WeakMap<target, Map<key, Set<effect>>>
. Pokud nebyl nalezen Set odběratelů efektů pro vlastnost (sledovanou poprvé), bude vytvořen. To je v krátkosti to, co funkce getSubscribersForProperty()
dělá. Pro jednoduchost se vyhneme jejím podrobnostem.
Uvnitř funkce trigger()
pro vlastnost opět vyhledáváme efekty odběratelů. Tentokrát je však voláme:
js
function trigger(target, key) {
const effects = getSubscribersForProperty(target, key)
effects.forEach((effect) => effect())
}
Nyní se vraťme k funkci whenDepsChange()
:
js
function whenDepsChange(update) {
const effect = () => {
activeEffect = effect
update()
activeEffect = null
}
effect()
}
Obaluje původní funkci update
v efektu, který nastavuje sám sebe jako aktuální aktivní efekt před spuštěním skutečné aktualizace. To umožňuje volání track()
během aktualizace pro nalezení aktuálního aktivního efektu.
V tomto bodě jsme vytvořili efekt, který automaticky sleduje své závislosti a znovu se spouští, kdykoli se změní některá závislost. Tento efekt nazýváme Reaktivní efekt (reactive effect).
Vue poskytuje API, které vám umožňuje reaktivní efekty vytvářet: watchEffect()
. Vlastně jste si možná všimli, že funguje docela podobně jako magická funkce whenDepsChange()
v příkladu. Nyní můžeme přepracovat původní příklad pomocí skutečných Vue API:
js
import { ref, watchEffect } from 'vue'
const A0 = ref(0)
const A1 = ref(1)
const A2 = ref()
watchEffect(() => {
// sleduje A0 a A1
A2.value = A0.value + A1.value
})
// spustí efekt
A0.value = 2
Použití reaktivního efektu pro změnu ref
není nejzajímavější použití - vlastně to bude názornější při použití computed proměnné:
js
import { ref, computed } from 'vue'
const A0 = ref(0)
const A1 = ref(1)
const A2 = computed(() => A0.value + A1.value)
A0.value = 2
Interně computed
spravuje svou neplatnost a nový výpočet pomocí reaktivního efektu.
Takže jaký je příklad běžného a užitečného reaktivního efektu? Například aktualizace DOM! Jednoduché „reaktivní vykreslování“ můžeme implementovat takto:
js
import { ref, watchEffect } from 'vue'
const count = ref(0)
watchEffect(() => {
document.body.innerHTML = `Počet je: ${count.value}`
})
// aktualizuje DOM
count.value++
Vlastně je to velmi podobné tomu, jak Vue komponenta udržuje synchronizaci svého stavu s DOM. Každá instance komponenty vytváří reaktivní efekt pro vykreslování a aktualizaci DOM. Samozřejmě, Vue komponenty používají mnohem efektivnější způsoby aktualizace DOM než innerHTML
. O nich se mluví v Mechanismu vykreslování.
Runtime vs. Compile-time reaktivita
Reaktivní systém Vue je převážně runtime-based: sledování a spouštění se provádí přímo v prohlížeči během spouštění kódu. Výhodou runtime reaktivity je, že může fungovat bez build fáze a je zde méně okrajových případů. Na druhou stranu je omezena syntaxí JavaScriptu, což vede k potřebě kontejnerů pro hodnoty jako jsou Vue refs.
Některé frameworky, například Svelte, se rozhodly tyto omezení překonat implementací reaktivity během kompilace. Analyzují a transformují kód, aby reaktivitu simulovaly. Kompilační krok umožňuje frameworku změnit sémantiku samotného JavaScriptu. Například implicitně vkládá kód, který provádí analýzu závislostí a spouštění efektů při přístupu k lokálně definovaným proměnným. Nevýhodou je, že takové transformace vyžadují build fázi a změna sémantiky JavaScriptu v podstatě vytváří jazyk, který vypadá jako JavaScript, ale kompiluje se do něčeho jiného.
Tým Vue tuto cestu zkoumal pomocí experimentální funkce nazvané transformace reaktivity, ale nakonec jsme se rozhodli, že by to pro projekt nebylo vhodné z důvodů uvedených zde.
Ladění reaktivity
Je skvělé, že reaktivní systém Vue sleduje závislosti automaticky, ale občas se může hodit zjistit, které závislosti jsou sledovány nebo co způsobuje překreslení komponenty.
Debugging Hooks komponenty
Pomocí lifecycle hooks onRenderTracked
a onRenderTriggered
můžeme ladit, které závislosti jsou během vykreslování komponenty používány a která závislost spouští aktualizaci. Oba hooks obdrží debug událost, která obsahuje informace o dané závislosti. Doporučuje se do těchto callbacků umístit příkaz debugger
, abyste mohli závislost interaktivně prozkoumat:
vue
<script setup>
import { onRenderTracked, onRenderTriggered } from 'vue'
onRenderTracked((event) => {
debugger
})
onRenderTriggered((event) => {
debugger
})
</script>
TIP
Debugging hooks komponent fungují pouze v režimu vývoje.
Objekty debug událostí mají následující typ:
ts
type DebuggerEvent = {
effect: ReactiveEffect
target: object
type:
| TrackOpTypes /* 'get' | 'has' | 'iterate' */
| TriggerOpTypes /* 'set' | 'add' | 'delete' | 'clear' */
key: any
newValue?: any
oldValue?: any
oldTarget?: Map<any, any> | Set<any>
}
Ladění computed proměnných
Computed proměnné můžeme ladit tak, že computed()
předáme druhý objekt s možnostmi onTrack
a onTrigger
:
onTrack
se zavolá, když je reaktivní vlastnost nebo ref sledován jako závislost.onTrigger
se zavolá, když je změnou závislosti spuštěn callback watcheru.
Oba callbacky obdrží debug události ve stejném formátu jako debug události komponent:
js
const plusOne = computed(() => count.value + 1, {
onTrack(e) {
// spuštěno, když je count.value sledováno jako závislost
debugger
},
onTrigger(e) {
// spuštěno, když je count.value změněno
debugger
}
})
// přístup k plusOne, mělo by spustit onTrack
console.log(plusOne.value)
// změna count.value, mělo by spustit onTrigger
count.value++
TIP
Možnosti onTrack
a onTrigger
pro computed proměnné fungují pouze v režimu vývoje.
Ladění watcherů
Podobně jako computed()
, watchery také podporují možnosti onTrack
a onTrigger
:
js
watch(source, callback, {
onTrack(e) {
debugger
},
onTrigger(e) {
debugger
}
})
watchEffect(callback, {
onTrack(e) {
debugger
},
onTrigger(e) {
debugger
}
})
TIP
Možnosti onTrack
a onTrigger
watcheru fungují pouze v režimu vývoje.
Integrace s externími systémy pro správu stavu
Reaktivní systém Vue funguje tak, že převádí běžné JavaScriptové objekty na hluboce reaktivní proxy. Hluboká konverze může být zbytečná a někdy i nežádoucí při integraci s externími systémy pro správu stavu (např. pokud i externí řešení používá Proxies).
Obecná myšlenka integrace reaktivního systému Vue s externím řešením pro správu stavu je uchovávat externí stav v shallowRef
. „Mělký“ ref je reaktivní pouze tehdy, když se přistupuje k jeho vlastnosti .value
– vnitřní hodnota zůstává nedotčena. Při změně externího stavu nahraďte hodnotu ref, aby se spustily aktualizace.
Neměnná data
Pokud implementujete funkci undo / redo, pravděpodobně chcete po každé úpravě uživatele vytvořit snímek stavu aplikace. Nicméně pokud je strom stavu rozsáhlý, měnitelný reaktivní systém Vue pro to není nejvhodnější, protože serializace celého stavového objektu při každé aktualizaci může být drahá z hlediska výkonu a paměťových nákladů.
Neměnné (immutable) datové struktury řeší tento problém tím, že stavové objekty nikdy nemění - místo toho vytvářejí nové objekty, které s těmi starými sdílejí stejné, nezměněné části. Existuje různé způsoby použití neměnných dat v JavaScriptu, ale s Vue doporučujeme použít Immer, protože vám umožňuje používat neměnná data a zachovává ergonomičtější, měnitelnou syntaxi.
Immer můžeme integrovat s Vue pomocí jednoduché composable:
js
import { produce } from 'immer'
import { shallowRef } from 'vue'
export function useImmer(baseState) {
const state = shallowRef(baseState)
const update = (updater) => {
state.value = produce(state.value, updater)
}
return [state, update]
}
Stavové automaty
Stavový automat (State machine) je model pro popis všech možných stavů, ve kterých se aplikace může nacházet, a všech možných způsobů, jak může přecházet z jednoho stavu do druhého. Ačkoli může být zbytečný pro jednoduché komponenty, může pomoci vytvořit robustnější a snadno spravovatelné složité stavové toky (state flows).
Jednou z nejpopulárnějších implementací stavových automatů v JavaScriptu je XState. Zde je composable, která s ním integruje:
js
import { createMachine, interpret } from 'xstate'
import { shallowRef } from 'vue'
export function useMachine(options) {
const machine = createMachine(options)
const state = shallowRef(machine.initialState)
const service = interpret(machine)
.onTransition((newState) => (state.value = newState))
.start()
const send = (event) => service.send(event)
return [state, send]
}
RxJS
RxJS je knihovna pro práci s asynchronními event streamy. Knihovna VueUse poskytuje add-on @vueuse/rxjs
na propojení RxJS proudů s reaktivním systémem Vue.
Propojení se signály
Celá řada dalších frameworků zavedla reaktivní prvky podobné refs z Composition API Vue pod termínem „signály“:
V jádru jsou signály stejným druhem prosté reaktivity (reactivity primitive) jako Vue refs. Jedná se o kontejner hodnot, který poskytuje sledování závislostí při přístupu a spouštění vedlejších efektů při změně. Toto paradigma založené na prosté reaktivitě není ve světě frontendu nijak zvlášť nový koncept: sahá až do implementací jako Knockout observables a Meteor Tracker z doby před více než deseti lety. Vue Options API a knihovna pro správu stavu Reactu MobX jsou také založeny na stejných principech, ale skrývají prostou reaktivitu za vlastnosti objektů.
I když to není nutná vlastnost, aby se něco kvalifikovalo jako signály, dnes se tento koncept často diskutuje ve spojitosti s modelem vykreslování, kde aktualizace probíhají prostřednictvím jemně granulovaných odběrů (fine-grained subscriptions). Vue v současné době díky použití virtuálního DOM pro dosažení podobných optimalizací spoléhá na překladač. Zkoumáme však také novou kompilační strategii inspirovanou frameworkem Solid (zvanou Vapor mode), která na virtuální DOM nespoléhá a více využívá vestavěný reaktivní systém Vue.
Kompromisy návrhu API
Návrh signálů v Preactu a Qwik je velmi podobný Vue shallowRef: všechny tři poskytují měnitelné rozhraní prostřednictvím vlastnosti .value
. Zaměříme se na diskuzi o signálech v Solidu a Angularu.
Solid Signály
Návrh API createSignal()
v Solidu zdůrazňuje oddělení čtení a zápisu. Signály jsou přístupné jako getter pouze pro čtení a samostatný setter:
js
const [count, setCount] = createSignal(0)
count() // přístup k hodnotě
setCount(1) // aktualizace hodnoty
Všimněte si, jak může být signál count
předán bez setteru. Tím se zajistí, že stav nemůže být nikdy měněn, pokud není setter explicitně vystaven. Zda tato záruka bezpečnosti ospravedlňuje více složitou syntaxi, může být závislé na požadavcích projektu a osobních preferencích. Pokud však tento styl API preferujete, můžete ho ve Vue snadno replikovat:
js
import { shallowRef, triggerRef } from 'vue'
export function createSignal(value, options) {
const r = shallowRef(value)
const get = () => r.value
const set = (v) => {
r.value = typeof v === 'function' ? v(r.value) : v
if (options?.equals === false) triggerRef(r)
}
return [get, set]
}
Angular Signály
Angular prochází některými zásadními změnami, jak se vzdává konceptu dirty-checking a představuje vlastní implementaci prosté reaktivity. API pro signály v Angularu vypadá takto:
js
const count = signal(0)
count() // přístup k hodnotě
count.set(1) // nastavení nové hodnoty
count.update((v) => v + 1) // aktualizace na základě předchozí hodnoty
Toto API opět můžeme snadno replikovat ve Vue:
js
import { shallowRef } from 'vue'
export function signal(initialValue) {
const r = shallowRef(initialValue)
const s = () => r.value
s.set = (value) => {
r.value = value
}
s.update = (updater) => {
r.value = updater(r.value)
}
return s
}
Ve srovnání s Vue refs poskytuje getter-based API styl Solidu a Angularu při použití ve Vue komponentách některé zajímavé kompromisy:
()
je o něco kratší než.value
, ale aktualizace hodnoty je rozsáhlejší.- Neexistuje ref-unwrapping: přístup k hodnotám vždy vyžaduje
()
. To zajišťuje všude konzistentní přístup k hodnotám. Také to znamená, že můžete předávat nezpracované signály jako vlastnosti (props) komponenty.
Zda se vám tyto API styly hodí, je do značné míry subjektivní. Naším cílem zde je demonstrovat podobnost a kompromisy mezi těmito různými návrhy API. Chceme také ukázat, že Vue je flexibilní: opravdu nejste uzamčeni do existujících API. Pokud je to nutné, můžete si vytvořit vlastní API pro reaktivitu, které lépe vyhovuje konkrétním potřebám.