В React компонентах, помимо props
, которые могут быть доступны только у первого потомка от родителя, есть контекст, который доступен у всех потомков родителя (объявившего контекст). Это означает, что как бы глубоко компонент не находился в дереве компонентов, он может иметь доступ к контексту, который мог быть создан на сколь угодно много ветвлений дерева выше (ближе к корню). На основе этого API работают все популярные библиотеки, которым необходимо иметь доступ к общим данным из любой глубины приложения: react-redux, react-mobx, react-router, styled-components (ThemeProvider).
Проблемы старого контекста
В старой версии данные можно было получить, но нельзя было подписаться на их обновление. Точнее их обновление приходило, как и props
, при перерендере родителя, но т.к. данные могут быть прокинуты между источником и потребителем контекста между большим (любым) количеством компонентов, которые могут остановить это обновление своим shouldComponentUpdate
, то надежной подписку, конечно, назвать было бы нельзя. Поэтому разработчикам библиотек, которым необходимо обновлять данные для использующих их компонентов приходилось продумывать механизмы обновления вручную, что на практике совсем не просто. Подробнее об этом можно почитать в статье или посмотреть доклад автора MobX
.
Новое API React.createContext
Кто-то мог воспринять новый контекст (React.createContext
— на русском) как замену redux
(или любого другого менеджера состояния), но это как сравнивать тёплое и мягкое. В действительности основная цель обновления контекста — это исправить проблему подписок на обновления и сделать это эффективно (подробности от Дэна Абрамова). Авторам библиотек не потребуется тратить силы и жертвовать объёмом бандла для собственной реализации подписок. Также обновлённый контекст предоставляет более простой и удобный интерфейс для его использования.
При этом можно отметить, что redux
имеет минимальное API для работы с состоянием: подписка и обновление, которое может быть заменено на использование state
(и setState
) из обычного React.Component
. Поэтому правильнее сказать, что новый контекст в каких-то задачах можно использовать вместо redux
, подразумевая, что вместо него будет использоваться состояние и обновление React.Component
, а вместо react-redux
— React.createContext
. При этом "in box" замены middleware
из redux
с использованием контекста нет, в этом случае можно воспользоваться сторонними библиотеками.
render-prop
Как можно заметить, новое API React.createContext
использует технику render-prop для связи с подписанными компонентами. Подробности реализации и примеры использования есть в официальной документации, мне же хотелось бы прояснить ключевые плюсы и минусы этого подхода:
- (+) Исключение коллизии имён при использовании нескольких подписок (
Consumer
). Классические HOC осуществляют слияниеprops
, и если у нас есть несколько HOC подряд, и у каких-то из них совпадают названия передаваемых параметров, то они будут перезатираться, и в конечном объектеprops
, который дойдёт до компонента, будет аргумент из последнего HOC. С render-prop эта проблема исчезает, т.к. с каждым передаваемым параметром подписки нужно работать индивидуально в передаваемой функции. - (-) "Сallback hell" и пересоздание функций или необходимость выносить части render, т.е. отрисовку отображения, в отдельные методы (что нарушает консистентность шаблона). Подробности — в официальной документации.
Если вам не нравится подход render-prop, и вы хотите использовать «старые добрые» HOC — вот простой пример, как это можно сделать с мемоизацией:
export const connect = selector => target => ({ children, ...props }) => {
let updateFromParent = true;
let cachedState = null;
let cachedComponent = null;
return (
<Consumer>
{context => {
const state = selector(context, props);
if (!updateFromParent && (state === cachedState || shallowCompare(state, cachedState))) {
updateFromParent = false;
return cachedComponent;
} else {
updateFromParent = false;
cachedState = state;
return (cachedComponent = React.createElement(target, { ...props, ...state }, children));
}
}}
</Consumer>
);
};
// example
// ComponentList.js
export const ComponentList = connect(context => ({ list: context.list }))(ComponentList_raw);
unstable_observedBits
информация взята из исходников и тестов React, а также из этой статьи
Публично об этом ещё не заявляли, и в официальной документации информации об этой части API нет (и, скорее всего, не будет), но, помимо вышесказанного, у React.createContext
есть второй аргумент, принимающий функцию, а у Consumer
есть параметр unstable_observedBits
, принимающий битовую маску сопоставления. Это аналогично shouldComponentUpdate
у React.Component
. Разберём подробнее.
Битовые маски
Битовые маски применяются очень давно, в частности, для сопоставления прав доступа в Linux. Суть заключается в том, что каждый бит в своей очерёдности на битовой маске отвечает за true
или false
по отношению к определённому правилу. Удобность работы с побитовыми масками заключается в том, что для обновления значения достаточно осуществить побитовую операцию оригинальной маски с маской правила, в которой для установки значения в true
необходимо применить "ИЛИ" — |
, где управляющий бит === 1
, а остальные — 0
, а для установки значения в false
необходимо применить "И" — &
, где управляющий бит === 0
, а остальные — 1
. Это может поначалу звучать сложно, но на практике это простой, наглядный, а главное, самый быстрый способ записи и управления диапазоном значений true
\ false
.
Использование
Битовая маска в описании и примерах ниже используется для отслеживания изменений в state
. Каждому значению state
должен соответствовать бит в битовой маске.
Второй аргумент React.createContext
принимает функцию, которая на вход получает предыдущее и обновлённое состояние, а на выходе должна вернуть обновлённую битовую маску. В свою очередь Consumer
принимает в unstable_observedBits
битовую маску, которая содержит биты положительных значений, отвечающих за отслеживаемые позиции state
. При поступлении изменений Consumer
сначала сравнивает обновлённую битовую маску с unstable_observedBits
, и если их побитовое сложение возвращает не 0
, то render-prop будет вызван, иначе — нет. Если второй аргумент React.createContext
и unstable_observedBits
у Consumer
не заданы — вызов render-prop будет происходить на любое изменение контекста.
Пример
const store = {
observedBits: {
foo: 0b01,
bar: 0b10
},
state: {
foo: 1,
bar: 1,
},
update(cb) {
this.state = cb(this.state);
}
};
const StoreContext = React.createContext(
store.state,
(prev, next) => {
let result = 0;
// поменялся `foo`
if (prev.foo !== next.foo) {
result |= store.observedBits.foo;
}
// поменялся `bar`
if (prev.bar !== next.bar) {
result |= store.observedBits.bar;
}
return result;
}
);
const Foo = () => (
<StoreContext.Consumer unstable_observedBits={store.observedBits.foo}>
{({foo, update}) => ( // если поменяется `bar`, этот код не выполнится
<button
onClick={() => update((state) => ({...state, foo: state.foo + 1}))}
>
increment "foo = {foo}"
</button>
)}
</StoreContext.Consumer>
);
Как ясно из названия параметров, данное API не стабильно и не стоит использовать его в проде.
create-subscription
Также в исходных кодах React появился пакет create-subscription
. Раньше для того, чтобы подписаться и как-то реагировать на внешние изменения и производить ререндер компонентов, необходимо было делать обёртку в виде класса React.Component
, в которой при поступлении уведомлений вызывать setState
или forceUpdate
. Для упрощения подписки теперь можно использовать более прозрачное API createSubscription
из пакета create-subscription
официального репозитория React.
Резюмируя
Обновление React 16.3 принесло множество интересных изменений и, безусловно, облегчит и повысит качество использования React и разработку вспомогательных библиотек для него.
Использование нового контекста можно посмотреть в интерактивно демонстрации:
Все вышеописанные технологии можно посмотреть в интерактивной демонстрации: