Содержание туториала по React
Установка переменной состояния поставит в очередь следующий рендеринг. Но иногда нужно выполнить несколько операций над значением перед постановкой в очередь следующего рендеринга. Для этого полезно понять, как React пакетно обновляет состояние (батчинг).
React батчит обновления состояний
Батчинг - это объединение обновлений в одну операцию.
В следующем примере, вы можете ожидать, что нажатие кнопки «+3» увеличит счетчик три раза, потому что он трижды вызывает setNumber(number + 1)
:
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
внутри обработчика событий первого рендеринга всегда равно 0
, независимо от того, сколько раз вы вызываете setNumber(1)
:
setNumber(0 + 1);
setNumber(0 + 1);
setNumber(0 + 1);
Но здесь есть еще один фактор, который нужно обсудить. React ждет, пока весь код в обработчиках событий будет запущен, прежде чем обрабатывать ваши обновления состояния. Вот почему повторный рендеринг происходит только после всех этих вызовов setNumber()
.
Это может напомнить вам официанта, принимающего заказ в ресторане. Официант не бежит на кухню при упоминании вашего первого блюда. Вместо этого они позволяют вам закончить свой заказ, позволяют вносить в него изменения и даже принимать заказы от других людей за столом.
Это позволяет вам обновлять несколько переменных состояния — даже из нескольких компонентов — без запуска слишком большого количества повторных рендерингов. Но это также означает, что пользовательский интерфейс не будет обновляться до тех пор, пока ваш обработчик событий и любой код в нем не завершится. Такое поведение, также известное как пакетная обработка (батчинг), позволяет вашему приложению React работать намного быстрее. Это также позволяет избежать сбивающих с толку «недоделанных» рендеров, в которых были обновлены только некоторые переменные.
React не объединяет несколько преднамеренных событий, таких как клики, — каждый клик обрабатывается отдельно. Будьте уверены, что React выполняет пакетную обработку только тогда, когда это в целом безопасно. Это гарантирует, что, например, если первый клик кнопки отключит форму, второй щелчок не отправит ее снова.
Обновление одной и той же переменной состояния несколько раз перед следующим рендерингом
Это необычный вариант использования, но если вы хотите обновить одну и ту же переменную состояния несколько раз перед следующим рендерингом, вместо передачи следующего значения состояния, такого как setNumber(number + 1)
, вы можете передать функцию, которая вычисляет следующее состояние. на основе предыдущего в очереди, например setNumber(n => n + 1)
. Это способ сказать React «сделать что-нибудь со значением состояния», а не просто заменить его.
Вот как это выглядит на практике:
import { useState } from 'react';
export default function Counter() {
const [number, setNumber] = useState(0);
return (
<>
<h1>{number}</h1>
<button
onClick={() => {
setNumber((n) => n + 1);
setNumber((n) => n + 1);
setNumber((n) => n + 1);
}}
>
+3
</button>
</>
);
}
Здесь n => n + 1
называется функцией обновления. Когда вы передаете его установщику состояния:
- React ставит эту функцию в очередь для обработки после выполнения всего остального кода в обработчике событий.
- Во время следующего рендеринга React проходит через очередь и выдает вам окончательное обновленное состояние.
setNumber((n) => n + 1);
setNumber((n) => n + 1);
setNumber((n) => n + 1);
Вот как React работает с этими строками кода при выполнении обработчика событий:
setNumber(n => n + 1)
: n => n + 1
— это функция. React добавляет его в очередь.
setNumber(n => n + 1)
: n => n + 1
— это функция. React добавляет его в очередь.
setNumber(n => n + 1)
: n => n + 1
— это функция. React добавляет его в очередь.
Когда вы вызываете useState
во время следующего рендеринга, React проходит через очередь. Предыдущее числовое состояние было 0
, так что это то, что React передает первой функции обновления в качестве аргумента n
. Затем React берет возвращаемое значение вашей предыдущей функции обновления и передает его следующему обновлению как n
и так далее:
запланированное обновление | n | возвращает |
---|
n => n + 1 | 0 | 0 + 1 = 1 |
n => n + 1 | 1 | 1 + 1 = 2 |
n => n + 1 | 2 | 2 + 1 = 3 |
React сохраняет 3
в качестве конечного результата и возвращает его из useState
.
Вот почему нажатие «+3» в приведенном выше примере корректно увеличивает значение на 3
.
Что произойдет, если вы обновите состояние после его замены
А как насчет этого обработчика событий? Как вы думаете, какое число будет в следующем рендере?
<button onClick={() => {
setNumber(number + 5);
setNumber(n => n + 1);
}}>
import { useState } from 'react';
export default function Counter() {
const [number, setNumber] = useState(0);
return (
<>
<h1>{number}</h1>
<button
onClick={() => {
setNumber(number + 5);
setNumber((n) => n + 1);
}}
>
Increase the number
</button>
</>
);
}
Вот что этот обработчик событий говорит React:
setNumber(number + 5)
: number
равно 0
, поэтому получаем setNumber(0 + 5)
. React добавляет в свою очередь «заменить на 5».
setNumber(n => n + 1)
: n => n + 1
— это функция обновления. React добавляет эту функцию в свою очередь.
Во время следующего рендера React проходит через очередь состояний:
запланированное обновление | n | возвращает |
---|
"заменить на 5" | 0 (не используется) | 5 |
n => n + 1 | 5 | 5 + 1 = 6 |
React сохраняет 6
в качестве конечного результата и возвращает его из useState
.
Вы могли заметить, что setState(x)
на самом деле работает как setState(n => x)
, но n
не используется.
Что произойдет, если вы замените состояние после его обновления
Давайте попробуем еще один пример. Как вы думаете, какое число будет в следующем рендере?
<button onClick={() => {
setNumber(number + 5);
setNumber(n => n + 1);
setNumber(42);
}}>
import { useState } from 'react';
export default function Counter() {
const [number, setNumber] = useState(0);
return (
<>
<h1>{number}</h1>
<button
onClick={() => {
setNumber(number + 5);
setNumber((n) => n + 1);
setNumber(42);
}}
>
Increase the number
</button>
</>
);
}
Вот как React работает с этими строками кода при выполнении этого обработчика событий:
setNumber(number + 5)
: number
равно 0
, поэтому setNumber(0 + 5)
. React добавляет в свою очередь «заменить на 5».
setNumber(n => n + 1)
: n => n + 1
— это функция обновления. React добавляет эту функцию в свою очередь.
setNumber(42)
: React добавляет в свою очередь «заменить на 42».
Во время следующего рендера React проходит через очередь состояний:
запланированное обновление | n | возвращает |
---|
"заменить на 5" | 0 (не используется) | 5 |
n => n + 1 | 5 | 5 + 1 = 6 |
"заменить на 42" | 6 (не используется) | 42 |
Затем React сохраняет 42
как окончательный результат и возвращает его из useState
.
Подводя итог, вот как вы можете думать о том, что вы передаете установщику состояния setNumber
:
- Функция обновления (например,
n => n + 1
) добавляется в очередь.
- Любое другое значение (например, число
5
) добавляет в очередь «заменить на 5», игнорируя то, что уже находится в очереди.
После завершения обработчика событий React запустит повторный рендеринг. Во время повторного рендеринга React обработает очередь. Функции обновления запускаются во время рендеринга, поэтому функции обновления должны быть чистыми и возвращать только результат. Не пытайтесь установить состояние внутри них или запустить другие побочные эффекты. В строгом режиме React дважды запускает каждую функцию обновления (но отбрасывает второй результат), чтобы помочь вам найти ошибки.
Соглашения об именах
Аргумент функции обновления принято называть первыми буквами соответствующей переменной состояния:
setEnabled((e) => !e);
setLastName((ln) => ln.reverse());
setFriendCount((fc) => fc * 2);
Если вы предпочитаете более подробный код, другим распространенным соглашением является повторение полного имени переменной состояния, например setEnabled(enabled => !enabled)
, или использование префикса, такого как setEnabled(prevEnabled => !prevEnabled)
.
Резюме
- Установка состояния не изменяет переменную в существующем рендеринге, но запрашивает новый рендеринг.
- React обрабатывает обновления состояния после завершения работы обработчиков событий. Это называется батчингом.
- Чтобы обновить некоторое состояние несколько раз в одном событии, вы можете использовать функцию обновления
setNumber(n => n + 1)
.
1. Что такое JSX?
- JSX — это расширение синтаксиса для JavaScript, обладающее всеми возможностями JavaScript.
- Вы можете внедрить любое выражение JavaScript в JSX, заключив его в фигурные скобки. После компиляции выражения JSX становятся обычными объектами JavaScript.
- Это означает, что вы можете использовать JSX внутри операторов
if
и циклов for
, назначать его переменным, принимать в качестве аргументов и возвращать из функций.
2. Каков эквивалент следующего кода с использованием React.createElement
?
const element = <h1 className="greeting">Hello, world!</h1>;
Эквивалент с использованием React.createElement
будет выглядеть следующим образом:
const element = React.createElement(
"h1",
{ className: "greeting" },
"Hello, world!"
);
3. Что такое Redux?
- Основная идея redux заключается в том, что все состояние приложения хранится в одном хранилище. Store (хранилище) - это просто JavaScript объект.
- Единственный способ изменить состояние — отправить действие (action) из вашего приложения, а затем написать редьюсеры для этих действий, которые изменяют состояние.
- Весь переход состояния хранится внутри редьюсеров и не должен иметь никаких побочных эффектов.
Подробнее о Redux можно узнать в бесплатном видео-курсе о Redux.
4. Что такое store в Redux?
Store — это JavaScript объект, который содержит состояние приложения. Наряду с этим он также имеет следующие обязанности:
- Разрешает доступ к состоянию через
getState()
.
- Позволяет обновлять состояние с помощью отправки действия
dispatch(action)
.
- Регистрирует слушателей через
subscribe(listener)
.
- Обрабатывает отмену регистрации слушателей с помощью функции, возвращаемой из
subscribe(listener)
.
5. Разница между action и reducer.
- Action (действие) — это простые JavaScript объекты.
- Они должны иметь тип, указывающий тип выполняемого действия.
- По сути, действия — это некоторые данные, которые отправляются из вашего приложения в хранилище.
- Редьюсер — это просто чистая функция, которая принимает предыдущее состояние и действие и возвращает обновленное состояние.
6. Для чего используется Redux Thunk?
- Redux Thunk — это промежуточное программное обеспечение (middleware), которое позволяет вам писать создателей действий (action creator), которые возвращают функцию вместо действия (action). Что такое action creators?
- Затем thunk можно использовать для задержки отправки действия, если выполняется определенное условие. Это позволяет вам обрабатывать асинхронную диспетчеризацию действий.
- Подробнее о Redux Thunk можно узнать в полном видео-курсе о Redux Thunk.
7. Напишите кастомный хук, который можно использовать для debounce’а ввода.
// Хук useDebounce
const useDebounce = (value, delay) => {
const [debouncedValue, setDebouncedValue] = useState(value);
useEffect(() => {
const timeout = setTimeout(() => {
setDebouncedValue(value);
}, delay);
return () => {
clearTimeout(timeout);
};
}, [value]);
return debouncedValue;
};
// Использование
const Counter = () => {
const [value, setValue] = useState(0);
const lastValue = useDebounce(value, 1000);
return (
<div>
<p>
Current Value: {value} | Debounced Value: {lastValue}
</p>
<button onClick={() => setValue(value + 1)}>Increment</button>
</div>
);
};
8. Напишите кастомный хук для копирования текста в буфер обмена.
// Хук useCopyToClipboard
function useCopyToClipboard(content) {
const [isCopied, setIsCopied] = useState(false);
const copy = useCallback(() => {
navigator.clipboard
.writeText(content)
.then(() => setIsCopied(true))
.then(() => setTimeout(() => setIsCopied(false), 1250))
.catch((err) => alert(err));
}, [content]);
return [isCopied, copy];
}
// Использование
export default function App() {
const [isCopied, copy] = useCopyToClipboard("Text to copy!");
return <button onClick={copy}>{isCopied ? "Copied!" : "Copy"}</button>;
}
9. Как использовать хук useId
для создания уникальных идентификаторов?
useId
не принимает никаких параметров.
useId
возвращает уникальную строку идентификатора, связанную с этим конкретным вызовом useId
в этом конкретном компоненте.
// Использование
import { useId } from "react";
const App = () => {
const id = useId();
return (
<form>
<label htmlFor={`email-${id}`}>Email</label>
<input type="text" id={`email-${id}`} name="email" />
<label htmlFor={`password-${id}`}>Password</label>
<input type="password" id={`password-${id}`} name="password" />
</form>
);
};
// Плохая практика - не стоит использовать в качестве key
const id = useId();
return posts.map((post) => <article key={id}>...</article>);
Подробности можно найти в статье Что за хук useId в React?.
10. Как проверить/валидировать пропсы в React?
Мы можем использовать пакет prop-types
Раньше, до React v15.5, это было частью самого React.
import PropTypes from "prop-types";
function MyComponent({ name }) {
return <div>Hello, {name}</div>;
}
MyComponent.propTypes = {
name: PropTypes.string,
};
export default MyComponent;
Еще один вариант - это добавить к проекту TypeScript.
11. Приведите практический пример компонента высшего порядка в React.
// Компонент высшего порядка
function WithLoading(Component) {
return function WihLoadingComponent({ isLoading, ...props }) {
if (!isLoading) return <Component {...props} />;
return <p>Please wait, fetching your data in no time...</p>;
};
}
export default WithLoading;
// Использование
import UserListComponent from "./UserListComponent.js"; // импорт компонента
import WithLoading from "./withLoading.js"; // импорт HOC
const ListWithLoading = WithLoading(UserListComponent); // обернем компонент в HOC
const App = () => {
const [loading, setLoading] = useState(true);
const [users, setUsers] = useState([]);
useEffect(() => {
// запрос данных
const dataFromApi = ["this is coming from API call", "don't show loader"];
// в это время загрузчик будет показан в HOC
// данные получены
setUsers([...dataFromApi]);
setLoading(false);
}, []);
return <ListWithLoading isLoading={loading} users={users} />;
};
12. Чем полезен хук useDeferredValue
?
useDeferredValue
— это React хук, который позволяет вам отложить обновление части пользовательского интерфейса.
- По сути, это позволяет вам выполнять технику debounce с меньшим количеством кода.
- Подробный разбор хука
useDeferredValue
можно прочитать в статье Хуки useTransition и useDeferredValue в ReactJS 18.
// Использование
import { useState, useDeferredValue } from "react";
// Компонент userList получает searchText для запроса списка пользователей
import UserList from "./UserList.js";
export default function App() {
const [searchText, setSearchText] = useState("");
// отправить searchText по умолчанию
const deferredQuery = useDeferredValue(searchText);
return (
<>
<label>
Search user:
<input
value={searchText}
onChange={(e) => setSearchText(e.target.value)}
/>
</label>
<div>
<UserList searchText={deferredQuery} />
</div>
</>
);
}
13. Как отследить клик за пределами компонента React?
export default function OutsideAlerter() {
const clickMeDivRef = useRef(null);
useEffect(() => {
const handleClickOutside = (event) => {
if (!ref?.current?.contains(event.target)) {
alert("You clicked outside of me!");
}
};
// Добавим прослушивание событий
document.addEventListener("mousedown", handleClickOutside);
return () => {
// Удалим прослушивание событий
document.removeEventListener("mousedown", handleClickOutside);
};
}, [clickMeDivRef]);
return <div ref={clickMeDivRef}>Clicked me?</div>;
}
14. Почему имена компонентов React должны начинаться с заглавных букв?
В JSX имена тегов нижнего регистра считаются тегами HTML. Однако имена тегов в нижнем регистре с точкой (аксессор свойства) не являются таковыми.
<person />
компилируется в React.createElement('person') (тег html)
- компилируется в
React.createElement(Person)
<obj.person />
компилируется в React.createElement(obj.person)
// Ошибка! Это компонент и должен начинаться с заглавной буквы
function person(props) {
// Верно! Это использование <div> верно, т.к. div это валидный элемент
return <div>{props.isLearning ? "Great!" : "Hello!"}</div>;
}
function App() {
// Ошибка! React считает <person /> тэгом HTML, т.к. Не с заглавной буквы
return <person isLearning={true} />;
}
// Верно! Это компонент и должен начинаться с заглавной буквы
function Person(props) {
// Верно! Это использование <div> верно, т.к. div это валидный элемент
return <div>{props.isLearning ? "Great!" : "Hello!"}</div>;
}
function App() {
// Верно! React знает, что <Person /> - это компонент, с заглавной буквы
return <Person isLearning={true} />;
}
15. В чем разница между npx и npm?
- npm — это менеджер пакетов, который можно использовать для установки пакетов node.js. npM - Manager.
- npx— это инструмент для выполнения пакетов node.js. npX - Execute.
Неважно, установили ли вы этот пакет глобально или локально. npx временно установит его и запустит. npm также может запускать пакеты, если вы настроите файл package.json
.
Поэтому, если вы хотите быстро проверить/запустить пакет без его установки - используйте npx.
create-react-app
— это npm пакет, который должен запускаться только один раз в жизненном цикле проекта. Следовательно, предпочтительнее использовать npx для установки и запуска за один шаг.
> npx create-react-app my-app
16. Как установить фокус на поле ввода после монтирования компонента в UI?
import React, { useEffect, useRef } from "react";
const SearchPage = () => {
const textInput = useRef(null);
useEffect(() => {
textInput.current.focus();
}, []);
return (
<div>
<input ref={textInput} type="text" />
</div>
);
};
Еще больше вопросов с собеседований