Стан як снепшот
Змінні стану можуть нагадувати звичайні змінні JavaScript, які можна зчитати та змінити. Проте стан поводиться радше як снепшот – окремий незмінний кадр. Задання йому значення не змінює наявну змінну стану, а натомість запускає повторний рендер.
You will learn
- Як задання стану запускає повторні рендери
- Коли та як оновлюється стан
- Чому стан не оновлюється відразу після задання значення
- Як обробники подій звертаються до “снепшоту” стану
Задання стану запускає рендери
Можна уявляти, що інтерфейс користувача змінюється безпосередньо внаслідок дії користувача, наприклад, клацання. У React це працює трохи інакше, ніж передбачає ця ментальна модель. На попередній сторінці ви побачили, що задання стану просить React про повторний рендер. Це означає, що необхідно оновити стан, щоб інтерфейс зреагував на подію.
У цьому прикладі, якщо натиснути “надіслати”, то setIsSent(true)
каже React виконати повторний рендер UI:
import { useState } from 'react'; export default function Form() { const [isSent, setIsSent] = useState(false); const [message, setMessage] = useState('Привіт!'); if (isSent) { return <h1>Ваше повідомлення — в дорозі!</h1> } return ( <form onSubmit={(e) => { e.preventDefault(); setIsSent(true); sendMessage(message); }}> <textarea placeholder="Повідомлення" value={message} onChange={e => setMessage(e.target.value)} /> <button type="submit">Надіслати</button> </form> ); } function sendMessage(message) { // ... }
Ось що відбувається, коли ви клацаєте кнопку:
- Виконується обробник події
onSubmit
. setIsSent(true)
задаєisSent
значенняtrue
і так додає до черги новий рендер.- React виконує повторний рендер компонента згідно з новим значенням
isSent
.
Погляньмо уважніше на взаємозв’язок між станом і рендерингом.
Кожен рендер отримує новий снепшот
“Рендеринг” означає, що React викликає ваш компонент, який є функцією. JSX, який ви повертаєте з цієї функції, — це як кадр UI в певну мить часу. Його пропси, обробники подій і локальні змінні обчислюються з використанням його стану в мить рендеру.
На відміну від світлини чи кадру кінострічки повернений “снепшот” UI інтерактивний. Він вміщає логіку штибу обробників подій, які визначають дії внаслідок введення користувачем. React оновлює екран до відповідності цьому снепшоту й приєднує обробники подій. Як наслідок, натискання кнопки запускає обробник клацання, заданий у вашому JSX.
Коли React виконує повторний рендер компонента, відбувається наступне:
- React знову викликає вашу функцію.
- Ваша функція повертає новий снепшот JSX.
- React потім оновлює екран до відповідності снепшоту, поверненого вашою функцією.
Illustrated by Rachel Lee Nabors
Стан, пам’ять компонента, не схожий на звичайну змінну, що зникає, коли відбувається повернення з функції. Стан фактично “живе” в самому React — неначе на поличці! — поза вашою функцією. Коли React викликає ваш компонент, він надає вам снепшот стану для конкретного поточного рендеру. Ваш компонент повертає снепшот UI зі свіжим набором пропсів і обробників подій у своєму JSX, обчисленим із використанням значень стану, взятих із цього рендеру!
Illustrated by Rachel Lee Nabors
Ось невеликий експеримент для демонстрації того, як це працює. У цьому прикладі можна очікувати, що клацання кнопки “+3” збільшить лічильник тричі, тому що це викликає setNumber(number + 1)
тричі.
Дивіться, що станеться, якщо клацнути кнопку “+3”:
import { useState } from 'react'; export default function Counter() { const [number, setNumber] = useState(0); return ( <> <h1>{number}</h1> <button onClick={() => { setNumber(number + 1); setNumber(number + 1); setNumber(number + 1); }}>+3</button> </> ) }
Зверніть увагу, що number
збільшується лише раз за одне клацання!
Задання стану змінює його лише для наступного рендеру. Під час першого рендеру number
був 0
. Саме тому в обробнику onClick
того конкретного рендеру значення number
все одно 0
навіть після виклику setNumber(number + 1)
:
<button onClick={() => {
setNumber(number + 1);
setNumber(number + 1);
setNumber(number + 1);
}}>+3</button>
Ось що обробник клацання цієї кнопки запитує в React:
setNumber(number + 1)
:number
—0
, тожsetNumber(0 + 1)
.- React готується змінити
number
на1
у наступному рендері.
- React готується змінити
setNumber(number + 1)
:number
—0
, тожsetNumber(0 + 1)
.- React готується змінити
number
на1
у наступному рендері.
- React готується змінити
setNumber(number + 1)
:number
—0
, тожsetNumber(0 + 1)
.- React готується змінити
number
на1
у наступному рендері.
- React готується змінити
Навіть попри те, що ви викликали setNumber(number + 1)
тричі, в обробнику подій поточного рендеру number
завжди дорівнює 0
, тож ви задаєте стану значення 1
тричі. Саме тому, коли завершується виконання вашого обробника помилок, React виконує повторний рендер компонента, де number
дорівнює 1
, а не 3
.
Також це можна візуалізувати, уявно замінюючи змінні стану їхніми значеннями в вашому коді. Оскільки змінна стану number
дорівнює 0
для поточного рендеру, обробник подій має наступний вигляд:
<button onClick={() => {
setNumber(0 + 1);
setNumber(0 + 1);
setNumber(0 + 1);
}}>+3</button>
Для наступного рендеру number
дорівнює 1
, тож для цього наступного рендеру обробник клацання має такий вигляд:
<button onClick={() => {
setNumber(1 + 1);
setNumber(1 + 1);
setNumber(1 + 1);
}}>+3</button>
Саме тому клацання кнопки знову задасть лічильнику значення 2
, після наступного клацання — 3
і так далі.
Стан протягом часу
Що ж, це було весело. Спробуйте вгадати, що покаже клацання цієї кнопки:
import { useState } from 'react'; export default function Counter() { const [number, setNumber] = useState(0); return ( <> <h1>{number}</h1> <button onClick={() => { setNumber(number + 5); alert(number); }}>+5</button> </> ) }
Якщо ви скористаєтесь методом заміни, описаним вище, то можете вгадати, що буде показано “0”:
setNumber(0 + 5);
alert(0);
А якщо поставити таймер на виведення повідомлення, щоб воно спрацювало лише після повторного рендеру компонента? Буде “0” чи “5”? Вгадайте!
import { useState } from 'react'; export default function Counter() { const [number, setNumber] = useState(0); return ( <> <h1>{number}</h1> <button onClick={() => { setNumber(number + 5); setTimeout(() => { alert(number); }, 3000); }}>+5</button> </> ) }
Здивовані? Якщо скористалися методом заміни, то бачите, який саме “снепшот” стану був переданий до повідомлення.
setNumber(0 + 5);
setTimeout(() => {
alert(0);
}, 3000);
Стан, що зберігається в React, може змінитися на час виведення повідомлення, але він був запланований з використанням снепшоту стану, актуального в момент взаємодії користувача з елементом!
Значення змінної стану ніколи не змінюється протягом одного рендеру, навіть якщо код обробника події є асинхронним. Всередині onClick
поточного рендеру значення number
усе одно залишається 0
навіть після виклику setNumber(number + 5)
. Її значення “зафіксувалося”, коли React “зробив снепшот” UI, викликавши ваш компонент.
Ось приклад, як це захищає обробники подій від помилок хронометражу. Нижче — форма, що надсилає повідомлення з п’ятисекундною затримкою. Уявіть такий сценарій:
- Ви натискаєте кнопку “Надіслати”, надсилаючи “Привіт” Анні.
- Перш ніж закінчиться п’ятисекундна затримка, ви змінюєте значення в полі “Кому” на “Богдан”.
Як гадаєте, що покаже alert
? Чи виведеться “Ви надіслали Привіт користувачу Анна”? Чи, можливо, “Ви надіслали Привіт користувачу Богдан”? Спробуйте вгадати на основі того, що знаєте, а потім перевірте:
import { useState } from 'react'; export default function Form() { const [to, setTo] = useState('Анна'); const [message, setMessage] = useState('Привіт'); function handleSubmit(e) { e.preventDefault(); setTimeout(() => { alert(`Ви надіслали ${message} користувачу ${to}`); }, 5000); } return ( <form onSubmit={handleSubmit}> <label> Кому:{' '} <select value={to} onChange={e => setTo(e.target.value)}> <option value="Анна">Анна</option> <option value="Богдан">Богдан</option> </select> </label> <textarea placeholder="Повідомлення" value={message} onChange={e => setMessage(e.target.value)} /> <button type="submit">Надіслати</button> </form> ); }
React тримає значення стану “зафіксованими” в межах обробників подій одного рендеру. Немає потреби перейматися тим, чи стан змінився, поки виконувався код.
Але що якщо хочеться зчитати найсвіжіший стан перед повторним рендером? Знадобиться скористатися функцією-оновлювачем стану, про яку мова піде на наступній сторінці!
Recap
- Задання стану спричиняє прохання про новий рендер.
- React зберігає стан поза вашим компонентом, неначе на поличці.
- Коли викликається
useState
, React віддає снепшот стану для конкретного поточного рендеру. - Змінні та обробники подій не “переживають” повторні рендери. Кожний рендер має власні обробники подій.
- Кожний рендер (а також функції в ньому) завжди “бачить” снепшот стану, який React віддав цьому конкретному рендеру.
- Можна уявно підставити стан в обробниках подій — подібного до того, як ми уявляємо JSX після рендеру.
- Обробники подій, створені в минулому, мають значення стану з тих рендерів, у яких вони створені.
Challenge 1 of 1: Реалізація світлофора
Ось компонент пішохідного світлофора, що перемикається, коли натискається кнопка:
import { useState } from 'react'; export default function TrafficLight() { const [walk, setWalk] = useState(true); function handleClick() { setWalk(!walk); } return ( <> <button onClick={handleClick}> Змінити на {walk ? 'Стійте' : 'Йдіть'} </button> <h1 style={{ color: walk ? 'darkgreen' : 'darkred' }}> {walk ? 'Йдіть' : 'Стійте'} </h1> </> ); }
Додайте alert
до обробника клацання. Коли світлофор світиться зеленим і каже “Йдіть”, натискання кнопки повинно видавати “Далі буде Стійте”. Коли світлофор світиться червоним і каже “Стійте”, натискання кнопки повинно видавати “Далі буде Йдіть”.
Чи важливий порядок: розташувати alert
до виклику setWalk
або навпаки?