В конце октября 2019 года вышел Redux Toolkit 1.0. Один из авторов проекта проекта и автор оригинальной публикации Марк Эриксон (Mark Erikson) рассказывает, как появился этот инструмент, делится целями создания Redux Starter Kit и объясняет, как удалось выполнить задуманное.
Краткая информация о Redux Toolkit:
- помогает быстро начать использовать Redux;
- упрощает работу с типичными задачами и кодом Redux;
- позволяет использовать лучшие практики Redux по умолчанию;
- предлагает решения, которые уменьшают недоверие к бойлерплейтам.
Содержание
Истоки
Появление Redux
Redux появился летом 2015 года. Он стал логическим завершением споров и войн вокруг Flux. В течение года с момента появления он обошёл другие имплементации Flux и де-факто стал стандартным инструментом для управления состояниями в React-приложениях.
Redux намеренно разрабатывался как расширяемый инструмент, и он стал именно таким инструментом. Вокруг проекта появилась целая экосистема аддонов: от Action/Reducer Generators до Store Persistence или десятков утилит для работы с неизменяемостью. Например, поскольку в Redux нет нативных инструментов для управления асинхронным поведением и побочными эффектами, появились десятки аддонов для работы с побочными эффектами.
Одним словом, если вам нужно что-то сделать с помощью Redux, с высокой долей вероятности уже есть аддон, который позволяет решить вашу задачу. В противном случае вы можете создать его самостоятельно.
Цикл хайпа
Практически все инструменты и технологии проходят через цикл хайпа. Когда появляется новый продукт, первые пользователи в него влюбляются и считают его серебряной пулей. Потом с новым продуктом знакомится больше людей. Они замечают ограничения и недостатки. Появляются негативные отзывы. Наконец, большинство пользователей находит лучшие способы применения технологии. Инструмент получает репутацию рабочего решения со своими преимуществами и недостатками.
Redux тоже прошёл через цикл хайпа. Разработчики быстро стали ассоциировать его с React. Появилась если не догма, то устойчивое мнение об обязательном использовании Redux там, где используется React. Некоторые senior-разработчики стали говорить новичкам, что они должны использовать Redux, если используют React.
В результате многие люди стали изучать Redux без осознания контекста. Новички часто не понимали, почему появился Redux, какие проблемы решает этот инструмент. Слепое использование любого инструмента — прямой путь к большим проблемам.
Слепое использование любого инструмента — прямой путь к большим проблемам.
Некоторые люди успешно использовали Redux и были довольны его возможностями. Другие заметили недостатки, причём не только реальные, но и придуманные.
Проблема сложности
Есть несколько факторов, которые в сумме делают использование Redux довольно сложным по сравнению с другими инструментами. Вот эти факторы:
- Redux намеренно добавляет уровень косвенности в идею управления состояниями. Как отмечает Дэн Абрамов, Redux не создавался в качестве самого производительного или самого простого инструмента управления мутациями. Он фокусируется на предсказуемости кода.
-
Общепринятые практики в Redux предполагают дополнительный уровень косвенности и использование «кода, который должен быть написан»: это создатели действий, константы типов действий, thunk’и, switch-выражения, неизменяемые обновления.
- Также для Redux написано много «правил», которые, впрочем, не соблюдаются самим Redux. Например, речь идёт о неизменяемости, сериализации, избегании побочных эффектов.
- JavaScript — мутабельный язык программирования, поэтому писать корректный иммутабельный код на нём сложно. Это связано как с необходимостью избегать случайных мутаций, так и с большим количеством кода, который нужно написать, чтобы управлять обновлениями без изменяемости.
- Redux всегда был инструментом с небольшим ядром, вокруг которого сформирована большая экосистема аддонов. Это позволяет пользователю выбирать настройки контейнера самостоятельно. С одной стороны, такой подход обеспечивает гибкость. С другой, даже простые задачи требуют от разработчика достаточно сложных действий, например, добавления мидлваров в асинхронную логику или настройку инструментов разработчика. Также это значит, что пользователь должен осознанно выбирать аддоны на ранних стадиях разработки, даже если приложение простое.
- Концепции функционального программирования сложны и незнакомы большинству пользователей, особенно людям с бэкграундом в ООП.
Как большинство ранних пользователей, первые пользователи Redux относились к категории продвинутых. Конфигурация, настройка и «ручная имплементация, которая позволяет контролировать все нюансы» были ключевыми преимуществами, с помощью которых инструмент привлекал новых пользователей. Тем не менее эти преимущества стали недостатками, когда в Redux пришло много новых пользователей с разным опытом и уровнем подготовки. Многие новички рассматривали паттерны и практики просто как бойлерплейт, с которым сложно работать. Поэтому они искали другие решения.
В дополнение к минимально необходимым, но сложным редьюсерам и диспетчеризации действий проблемой стала случайная сложность. Например, документация Redux рекомендует организовывать код в файлы и директории по действиям, константам, редьюсерам и контейнерам. Есть веские причины распределять код в файлы по типам. В то же время нет обязательных требований писать действия, редьюсеры и константы в отдельных файлах или директориях. Большинство пользователей читали документацию и выполняли рекомендации, поэтому организация кода по действиям, константам, редьюсерам и контейнером стала стандартным паттерном.
Похожий пример: написание типов действий как констант ( const ADD_TODO = "ADD_TODO" ) или наличие функций, создающих действие для каждого типа, также не требуется в обязательном порядке. Но эти подходы вошли в рекомендованные паттерны, а пользователи активно следуют этим рекомендациям.
Необходимость выбирать аддоны часто вызывала сложности у пользователей. Болевой точкой стало большое количество библиотек с побочными эффектами, в каждой из которых используется своя терминология. Уже через год после появления Redux стало понятно, что побочные эффекты — большая проблема.
Экосистема держалась на thunk, сагах и наблюдаемых объектах. Одним из самых распространённых вопросов у пользователей стал «как мне выбрать мидлвар для работы с побочными эффектами». Кстати, thunk изначально были в ядре Redux, но их убрали из него ради дополнительной гибкости. Очевидно, что пользователи нуждались в «официальных» инструментах для работы с побочными эффектами.
В поисках решения
Автор оригинальной публикации погрузился в работу над Redux в 2016 году, когда в качестве волонтёра писал раздел FAQ на официальном сайте проекта. Дэн Абрамов выдал автору и Тиму Дорру карт-бланш, и автор начал отвечать на вопросы пользователей на разных сайтах и форумах.
К весне 2017 года автор заметил изменения в сообществе: большинство пользователей не понимали, для чего создавался Redux, а также почему существуют типичные паттерны применения этого инструмента. Автор попытался исправить это с помощью нескольких подробных публикаций: «Дао Redux, часть 1: реализация и намерение» и «Дао Redux, часть 2: практика и философия». Эти статьи рассказывают, для чего создан Redux и как его планировалось использовать.
Примерно в то же время автор открыл issue в репозитории Redux. В нём поднимались такие вопросы:
- Каким должно быть идиоматическое использование Redux с условием ухода от бойлерплейта? Как мы можем реагировать на подобные жалобы?
- Какие абстракции можно создать, чтобы упростить изучение и использование Redux без необходимости прятать этот инструмент?
В первом ответе Джастин Фалькон (Justin Falcone) предположил следующее:
- Нужен официальный пакет redux-preset с redux, react-redux, redux-thunk.
- Нужны создатели редьюсеров, например, createReducer({[actionType]: (state, payload) => state}, initState).
Эти предположения стали основой будущего Redux Toolkit.
В треде появилось более сотни комментариев. В итоге некоторые пользователи просто рекламировали свои библиотеки для работы с абстракциями. В середине треда Дэн Абрамов оставил свои пожелания:
- Подмножество Flow/TypeScript, с которым проще работать, чем с чистым Redux.
- Уход от выделения констант, просто сделайте строковые литералы более безопасными.
- Сохранение независимости от React и других библиотек, но с упрощением использования существующих зависимостей, например, можно предоставить реализации mapStateToProps.
- Сохранение принципов Redux: сериализуемые действия, путешествия во времени и горячая перезагрузка должны работать, журнал действий должен иметь смысл.
- Разделение кода простым способом из коробки.
- Поддержка использования редьюсеров с селекторами, а также создание условий, в которых они не будут казаться ужасными вместе (подумайте о парах редьюсер-селектор, которые легко писать).
- Вместо совмещения создателей действий с редьюсерами, стоит полностью уйти от создателей действий.
- Принять разумные значения производительности по умолчанию, чтобы мемоизация через Reselect просто работала без необходимости самостоятельно писать код.
- Создание встроенных помощников для индексации, нормализации, коллекций.
- Поддержка встроенного тестируемого асинхронного потока.
- Инструмент должен быть брендированным и позиционироваться в качестве официального.
Начало процесса
В феврале 2018 Ник Маккерди (Nick McCurdy) открыл issue, в котором предложил внедрить заморозку состояния на стадии разработки. Автор оригинальной публикации написал в треде такой комментарий:
«Я уже писал, что хочу создать стартовый пакет, в котором будет официально рекомендованный командой Redux и простой набор инструментов. Я посмотрел десятки вариантов, включая вариант Create React App + Redux, но пока мы ничего не одобрили.
Настоящая проблема в том, что ядро Redux из коробки вообще не имеет установленных конфигураций. Мы не предполагаем, какой мидлвар будет использовать разработчик, и будет ли он вообще его использовать. Поэтому мы не можем просто добавить мидлвар в ядро, потому что в ядре из коробки нет установленных конфигураций».
В ответ Тим Дорр открыл issue и написал следующее:
«Похоже, для успешного начала работы с библиотекой было бы полезно сделать стартеркит с ядром Redux, в которым также должны быть несколько общих мидлваров, несколько расширителей (enhancers).
В моём представлении в стартовом пакете должна быть одна из библиотек редьюсеров, популярные мидлвары, например, thunk’и и саги, полезные инструменты разработки. Можно сделать подмножество пакетов для тех, кто использует React.
Не думаю, что нужно делать что-то похожее на Create React App. Но нам нужно что-то, что из коробки даёт все инструменты, нужные для начала работы».
После этого на Reactiflux началась продуктивная дискуссия с участием автора оригинальной публикации, Ника Маккерди и Гарри Волфа (Harry Wolff). По её итогам Гарри Волф написал, что могло бы быть в стартовом пакете Redux:
- обёртка для createStore с хорошим набором дефолтных инструментов, например, простой способ добавления мидлваров, инструменты разработчика Redux;
- встроенный хелпер для работы с неизменяемостью (Immer?);
- встроенный createReducer;
- создатель действий, который совместим с FSA;
- поддержка асинхронных действий, поддержка побочных эффектов — Flux, сага, что-то ещё?
На следующий день автор этой статьи опубликовал пример реализации configureStore и createReducer. Ещё через два дня автор опубликовал Redux Starter Kit как экспериментальный пакет.
Проектирование инструментов
Первые имплементации и нейминг
Работа над Redux Toolkit продолжалась в течение нескольких следующих месяцев. Ник добавил первые юнит-тесты, интеграцию с CI, включил инструменты разработчика Redux в configureStore. Автор статьи добавил реэкспорт из Redux и библиотеку selectorator, а также улучшил настройки мидлваров. После высокой активности с марта по май 2018 наступил период затишья, который продолжался несколько месяцев.
Автор публикации ощутил выгорание летом 2018 года из-за высокой нагрузки. После обдумывания ситуации он решил сосредоточиться на обязанностях мейнтейнера проекта. Это предполагало в числе других активностей популяризацию Redux Starter Kit.
Выбранное название казалось отвлечённым. Именно поэтому автор статьи опубликовал инструмент как личный пакет. В августе 2018 года он открыл issue, в котором начал дискуссию о финальном названии инструмента. В числе прочего предлагалось использовать префикс @redux.
Дискуссия продолжалась до середины октября 2018 года. Один из участников предложил вернуться к простому варианту redux-starter-kit. Проблема была в том, что кто-то уже опубликовал пакет с названием redux-toolkit на npm. Автор статьи на всякий случай обратился к владельцу пакета с ником @shotak и спросил, не может ли он подарить имя redux-toolkit. К удивлению автора, человек ответил в течение нескольких часов и согласился подарить имя. После этого автор опубликовал первую официальную версию инструмента.
Работа над функциональностью
Функции createReducer и createAction уже были полезными. Но автор статьи хотел найти более простые инструменты.
Команда обсуждала использование хелперов из пакетов типа redux-utilities.
Автор публикации видел десятки подобных библиотек. Одна из библиотек, которая привлекла внимание автора — autodux Эрика Эллиотта. Она позволяет пользователям автоматически генерировать создателей действий и типы действий на основе объекта с редьюсерами и префикса с именем.
Шон Ванг (Shawn Wang) оценил autodux и несколько других пакетов. Вот выдержка:
«Использование autodux даёт мощный набор создателей, селекторов и редьюсеров из коробки. Поэтому вы можете писать действия для бизнес-логики без дополнительных усилий. Мне нравится autodux».
В конце концов autodux стал источником вдохновения при создании самой полезной части Redux Toolkit: функции createSlice.
Эрик Боуэр (Eric Bower) попробовал портировать autodaz в createSlice. Пулреквест застопорился. Автор статьи узнал о библиотеке robodux, в которой используется Immer. Он понял, что Эрик Боуэр решил сделать собственную версию библиотеки на TypeScript.
После обсуждения Боуэр согласился вернуть код в Redux Toolkit. Автор скомпилировал код библиотеки в чистый JavaScript и включил его в стартеркит.
Создание дорожной карты
В начале декабря 2018 года с помощью Docusaurus была опубликована документация Redux Toolkit.
Автор статьи добавил в стартеркит дефолтные мидлвары для определения мутаций и несериализируемых значений, а также возможность трассировки стека.
В конце января 2019 года автор статьи открыл issue, в котором предложил обсудить дорожную карту к выпуску redux Starter Kit 1.0, а также начал работать над способностью слайсов слушать другие действия. Дэнис Вашингтон (Denis Washington) прокомментировал этот пулреквест. Он предложил обсудить целесообразность использования createSlice.
В ответ автор публикации написал большой комментарий. Вот выдержки:
Цели
Согласно README, в числе прочих источниками вдохновения для создателей Redux Toolkit являются Create-React-App и Apollo-Boost. Это пакеты, в которых есть дефолтные инструменты и абстракции над более сложными настраиваемыми инструментами. Цель существования этих пакетов — обеспечить быстрый старт при работе с библиотекой без необходимости погружаться в детали и принимать сложные решения. В то же время эти пакеты не привязывают разработчика к дефолтным инструментам и настройкам. Они очень полезны для новичков, которые не знают о всех доступных вариантах и о «правильных» выборах. Также пакеты полезны для опытных разработчиков, которым нужен простой инструмент, не требующий настройки окружения.
Я хочу, чтобы Redux Toolkit стал таким же инструментом для Redux.
Быстрый старт
Я хочу, чтобы люди начинали работать с Redux как можно быстрее.
Упрощение основных способов применения и уход от бойлерплейтов
Я хочу убрать острые углы:
- сократить код где это возможно;
- предложить функции, которые выполняют очевидные действия, которые пользователи привыкли делать руками;
- посмотреть, как люди используют Redux и как они хотят его использовать, и предложить «официальное» решение на основе предпочтений пользователей.
Redux Toolkit должен вести пользователя к правильным настройкам, а также автоматически защищать от распространённых ошибок или предупреждать о них.
Отсутствие блокировки
Когда кто-то начинает делать приложение на базе Redux Toolkit, я не хочу забирать у него возможность сделать что-то вручную в начале или в процессе работы.
С другой стороны, люди должны иметь возможность добавлять Redux Toolkit в существующие приложения. В таких случаях у разработчиков должна быть возможность внедрять стартеркит постепенно без необходимости переписывать код с нуля.
Стать очевидным дефолтным способом использования Redux
В долгосрочной перспективе автор видит Redux Toolkit очевидным дефолтным способом использования Redux для большинства разработчиков. В качестве примера можно посмотреть на Create React App, который стал дефолтным очевидным способом использования React.
Отказ от попыток охватить все способы использования Redux, фокусировка на типичных подходах
Есть много способов использования Redux. А в ядре Redux есть минимальный набор инструментов. Именно поэтому существуют тысячи библиотек и аддонов, который охватывают разные кейсы применения Redux. Стартеркит не может и не собирается охватывать все способы применения Redux.
Разумная поддержка TypeScript
Автор хочет поддерживать специалистов, которые пишут на TypeScript, как и других специалистов, которые используют Redux специфическим способом. Поэтому в Redux Toolkit должна быть хорошая типизация. В то же время автор не хочет, чтобы «лучшая типизация» стала врагом «хорошей функциональности».
Работа с API
Параллельно с этой дискуссией Дэнис Вашингтон конвертировал базу Redux Toolkit в TypeScript.
Автор публикации потратил несколько месяцев на работу с дорожной картой React-Redux 7. Потом он работал с дизайном хуков API.
Одновременно команда обсуждала разные вопросы, включая расширение возможностей createSlice. Рассматривалась функция combineSlices, а также автогенерируемый селектор функциональности.
К лету 2019 года автор завершил работу над React-Redux и начал писать документацию Redux Toolkit. В начале сентября она была готова.
Тем временем Ленц Вебер (Lenz Weber) предложил несколько улучшений, включая внедрение колбэка prepare в createAction, а также улучшение читабельности определений типов.
С этого момента автор начал «допиливать» API Redux Toolkit. Появилась возможность кастомизации дефолтных мидлваров, а также включение редьюсеров в createSlice.
Было ещё много идей, но автор решил, что их можно реализовать после релиза. После этого официально вышел Redux Toolkit 1.0.
Будущее Redux Toolkit
Обзор
Оглядываясь на проделанную работу, можно сказать, что создателям удалось выполнить основные запланированные задачи. Redux Toolkit стал инструментом с такими характеристиками и возможностями:
- официальный пакет, в котором реализованы лучшие практики;
- значительно упрощает работу с Redux;
- сохраняет основные принципы Redux;
- стартеркит можно добавить в проект на любой стадии реализации;
- полезный инструмент как для новичков, так и для опытных разработчиков;
- поддерживает TypeScript без увеличения сложности разработки.
Автор активно рекламировал Redux Toolkit целый год. Ссылка на этот инструмент появилась на главной странице документации Redux. Также инструмент рекомендуется в разделе документации Configuring Your Store.
Судя по статистике загрузки пакета, Redux Toolkit оказался востребованным инструментом. Его устанавливают более 100 тыс. раз в месяц, и график количества загрузок показывает рост. Автор получил много положительных откликов о Redux Toolkit.
Следующие шаги
В настоящее время планируется масштабная работа по модернизации документации Redux. В рамках модернизации будет созданы обучающие материалы по Redux Toolkit. Также информация о стартерките будет добавлена в гайды и туториалы. Пользователи узнают о простых способах работы с Redux, например, об избегании использования констант действия или об использовании утиного паттерна при организации логики. Возможно, в Redux Toolkit добавят возможность определять асинхронные побочные эффекты в слайсах.
Адаптированный перевод статьи Idiomatic Redux: Redux Toolkit 1.0 by Mark Erikson. Мнение администрации «Хекслета» может не совпадать с мнением автора оригинальной публикации.
<!DOCTYPE html>
<html class="h-100" data-bs-theme="light" data-mantine-color-scheme="light" lang="ru" prefix="og: https://ogp.me/ns#">
<head>
<meta content="width=device-width, initial-scale=1.0" name="viewport">
<meta content="IE=Edge" http-equiv="X-UA-Compatible">
<link crossorigin="true" href="https://cdn.hexlet.io" rel="preconnect">
<link href="https://mc.yandex.ru" rel="preconnect">
<meta content="aa2vrdtq64dub8knuf83lwywit311w" name="facebook-domain-verification">
<link href="/favicon.ico" rel="icon" sizes="any">
<link href="/favicon.svg" rel="icon" type="image/svg+xml">
<link href="/apple-touch-icon.png" rel="apple-touch-icon">
<link href="/manifest.webmanifest" rel="manifest">
<script>
//<![CDATA[
window.gon={};gon.ym_counter="25559621";gon.is_bot=true;gon.applications={};gon.current_user={"id":null,"last_viewed_notification_id":null,"email":null,"state":null,"first_name":"","last_name":"","created_at":"2026-02-26 20:39:27 UTC","current_program":null,"current_team":null,"full_name":"","guest":true,"can_use_paid_features":false,"is_hexlet_employee":false,"sanitized_phone_number":"","can_subscribe":true,"can_renew_education":false};gon.token="cH7i7piHw3rXkvCehrUjB4wYftoAdi6F_CXJ5RENhwSfrynZavluGmHR1AaKutNwTBFTcAhB0CdBxVOxQwpgag";gon.locale="ru";gon.language="ru";gon.theme="light";gon.rails_env="production";gon.mobile=false;gon.google={"analytics_key":"UA-1360700-51","optimize_key":"GTM-5QDVFPF"};gon.captcha={"google_v3_site_key":"6LenGbgZAAAAAM7HbrDbn5JlizCSzPcS767c9vaY","yandex_site_key":"ysc1_Vyob5ZPPUdPBsu0ykt8bVFdzsfpoVjQChLGl2b4g19647a89","verification_failed":null};gon.social_signin=false;gon.typoreporter_google_form_id="1FAIpQLSeibfGq-KvWQ2Fyru-zkFFRVTLBuzXAHAoEyN1p49FtDmNoNA";
//]]>
</script>
<meta charset="utf-8">
<title>Идиоматический Redux: Redux Toolkit 1.0</title>
<meta name="description" content="В конце октября 2019 года вышел Redux Toolkit 1.0. Один из авторов проекта проекта и автор оригинальной публикации Марк Эриксон (Mark Erikson) рассказывает, как появился этот инструмент, делится целями создания Redux Starter Kit и объясняет, как удалось выполнить задуманное.">
<link rel="canonical" href="https://ru.hexlet.io/blog/posts/idiomaticheskiy-redux-redux-starter-kit-1-0">
<meta property="og:title" content="Идиоматический Redux: Redux Toolkit 1.0">
<meta property="og:description" content="В конце октября 2019 года вышел Redux Toolkit 1.0. Один из авторов проекта проекта и автор оригинальной публикации Марк Эриксон (Mark Erikson) рассказывает, как появился этот инструмент, делится целями создания Redux Starter Kit и объясняет, как удалось выполнить задуманное. ">
<meta property="og:image" content="https://ru.hexlet.io/vite/assets/blog_post-7eTyeLLt.webp">
<meta name="csrf-param" content="authenticity_token" />
<meta name="csrf-token" content="3Xb8pvjlsDs3vBz02QUOr4t0OQyYTGMj0VoPhqRE8XIypzeRCpsdW4H_OGzVCv7YS30UppB7nYFsupXS9kMWHA" />
<script src="/vite/assets/inertia-DfXos102.js" crossorigin="anonymous" type="module"></script><link rel="modulepreload" href="/vite/assets/chunk-DsPFFUou.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/preload-helper-BJ4cLWpC.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/init-BrRXra1y.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/ahoy-DrlRQ-1D.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/analytics-cb8xch9l.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/ErrorFallbackBlock-naDSYSy9.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Surface-DL2bpZA-.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/gon-D3e4yh1x.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/mantine-CGMYrt2Y.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/utils-DRqSHbQE.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/routes-CCH8ilKF.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/extends-C-EagtpE.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/inheritsLoose-BBd-DCVI.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/objectWithoutPropertiesLoose-DRHXDhjp.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/index.esm-DAqKOkZ0.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Button-CGPUux8l.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/CloseButton-D1euiPao.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Group-BX48WcuU.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Loader-BQEY8g6v.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Modal-Cy3HByv7.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/OptionalPortal-1Hza5P2w.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Stack-CtjJzfw4.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Textarea-Ck64llAy.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Box-B5-OOzBf.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/DirectionProvider-Dc9zdUke.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/events-DJQOhap0.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/use-reduced-motion-D2owz4wa.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/use-disclosure-zKtK5W1r.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/use-hotkeys-Cnc_Rwkb.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/random-id-DOQyszCZ.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/notifications.store-C-3AFSMn.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/exports-C_MrNx_T.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/axios-BEvgo0ym.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/dayjs.min-BkKovM-s.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/i18next-BlSq9s7B.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/client-U9M77rxp.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/react-dom-DaLxUz_h.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/useTranslation-Bx1Cdrkz.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/compiler-runtime-6XxiPFnt.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/jsx-runtime-CwjcCKJi.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/react-CkL4ZRHB.js" as="script" crossorigin="anonymous">
<link rel="stylesheet" href="/vite/assets/application-BqhCP46M.js" />
<script src="/vite/assets/application-Df9RExpe.js" crossorigin="anonymous" type="module"></script><link rel="modulepreload" href="/vite/assets/chunk-DsPFFUou.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/autocomplete-VMNbxKGl.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/routes-CCH8ilKF.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/createPopper-C3aM9r1M.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/js.cookie-D1-O8zkX.js" as="script" crossorigin="anonymous"><link rel="stylesheet" href="/vite/assets/application-C8HjmMaq.css" media="screen" />
<script>
window.ym = function(){(ym.a=ym.a||[]).push(arguments)};
window.addEventListener('load', function() {
setTimeout(function() {
ym.l = 1*new Date();
ym(window.gon.ym_counter, "init", {
clickmap: true,
trackLinks: true,
accurateTrackBounce: true,
webvisor: true
});
// Загружаем скрипт
var k = document.createElement('script');
k.async = 1;
k.src = 'https://mc.yandex.ru/metrika/tag.js';
document.head.appendChild(k);
ym(window.gon.ym_counter, 'getClientID', function(clientID) {
window.ymClientId = clientID;
});
}, 1500);
});
</script>
<!-- Google Tag Manager - deferred -->
<script>
// dataLayer stub сразу — пуши работают до загрузки скрипта
window.dataLayer = window.dataLayer || [];
// Сам скрипт — отложенно после load
window.addEventListener('load', function() {
setTimeout(function() {
dataLayer.push({'gtm.start': new Date().getTime(), event: 'gtm.js'});
var j = document.createElement('script');
j.async = true;
j.src = 'https://www.googletagmanager.com/gtm.js?id=GTM-WK88TH';
document.head.appendChild(j);
}, 1500);
});
</script>
<!-- End Google Tag Manager -->
</head>
<body>
<noscript>
<div>
<img alt="" src="https://mc.yandex.ru/watch/25559621" style="position:absolute; left:-9999px;">
</div>
</noscript>
<header class="sticky-top bg-body">
<nav class="navbar navbar-expand-lg">
<div class="container-xxl">
<a class="navbar-brand" href="/"><img alt="Логотип Хекслета" height="24" src="https://ru.hexlet.io/vite/assets/logo_ru_light-BpiEA1LT.svg" width="96">
</a><button aria-controls="collapsable" aria-expanded="false" aria-label="Меню" class="navbar-toggler border-0 mb-0 mt-1" data-bs-target="#collapsable" data-bs-toggle="collapse">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse" id="collapsable">
<ul class="navbar-nav mb-lg-0 mt-lg-1">
<li class="nav-item dropdown">
<button aria-haspopup class="btn nav-link" data-bs-toggle="dropdown" type="button">
Все курсы
<span class="bi bi-chevron-down align-middle ms-1"></span>
</button>
<ul class="dropdown-menu">
<li>
<a class="dropdown-item d-flex py-2" href="/courses"><div class="fw-bold me-auto">Все что есть</div>
<div class="text-muted">117</div>
</a></li>
<li>
<hr class="dropdown-divider">
</li>
<li class="dropdown-item">
<b>Популярные категории</b>
</li>
<li>
<a class="dropdown-item py-2" href="/courses_devops">Курсы по DevOps
</a></li>
<li>
<a class="dropdown-item py-2" href="/courses_data_analytics">Курсы по аналитике данных
</a></li>
<li>
<a class="dropdown-item py-2" href="/courses_programming">Курсы по программированию
</a></li>
<li>
<a class="dropdown-item py-2" href="/courses_testing">Курсы по тестированию
</a></li>
<li>
<hr class="dropdown-divider">
</li>
<li class="dropdown-item">
<b>Популярные курсы</b>
</li>
<li>
<a class="dropdown-item py-2" href="/programs/devops-engineer-from-scratch">DevOps-инженер с нуля
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/go">Go-разработчик
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/java">Java-разработчик
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/python">Python-разработчик
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/qa-auto-engineer-java">Автоматизатор тестирования на Java
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/data-analytics">Аналитик данных
</a></li>
<li>
<a class="dropdown-item py-2" href="/programs/frontend">Фронтенд-разработчик
</a></li>
</ul>
</li>
<li class="nav-item dropdown">
<button aria-haspopup class="btn nav-link" data-bs-toggle="dropdown" type="button">
О Хекслете
<span class="bi bi-chevron-down align-middle"></span>
</button>
<ul class="dropdown-menu bg-body">
<li>
<a class="dropdown-item py-2" href="/pages/about">О нас
</a></li>
<li>
<a class="dropdown-item py-2" href="/blog">Блог
</a></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://special.hexlet.io/hse-research" role="button">Результаты (Исследование)
</span></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://career.hexlet.io" role="button">Хекслет Карьера
</span></li>
<li>
<a class="dropdown-item py-2" href="/testimonials">Отзывы студентов
</a></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://t.me/hexlet_help_bot" role="button">Поддержка (В ТГ)
</span></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://special.hexlet.io/referal-program/?promo_creative=priglasite-druzei&promo_name=referal-program&promo_position=promo_position&promo_start=010724&promo_type=link" role="button">Реферальная программа
</span></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://special.hexlet.io/certificate" role="button">Подарочные сертификаты
</span></li>
<li>
<span class="dropdown-item py-2 external-link" data-href="https://hh.ru/employer/4307094" role="button">Вакансии
</span></li>
<li>
<span class="dropdown-item d-flex external-link" rel="noopener noreferrer nofollow" data-href="https://b2b.hexlet.io" data-target="_blank" role="button">Компаниям
</span></li>
<li>
<span class="dropdown-item d-flex external-link" rel="noopener noreferrer nofollow" data-href="https://hexly.ru/" data-target="_blank" role="button">Колледж
</span></li>
<li>
<span class="dropdown-item d-flex external-link" rel="noopener noreferrer nofollow" data-href="https://hexlyschool.ru/" data-target="_blank" role="button">Частная школа
</span></li>
</ul>
</li>
<li><a class="nav-link" href="/subscription/new">Подписка</a></li>
</ul>
<ul class="navbar-nav flex-lg-row align-items-lg-center gap-2 ms-auto">
<li>
<a class="nav-link" aria-label="Переключить тему" href="/theme/switch?new_theme=dark"><span aria-hidden="true" class="bi bi-moon"></span>
</a></li>
<li>
<span data-target="_self" class="nav-link external-link" data-href="/u/new" role="button"><span>Регистрация</span>
</span></li>
<li>
<span data-target="_self" class="nav-link external-link" data-href="https://ru.hexlet.io/session/new" role="button"><span>Вход</span>
</span></li>
</ul>
</div>
</div>
</nav>
</header>
<div class="x-container-xxxl">
</div>
<main class="mb-6 min-vh-100 h-100">
<link rel="preload" as="image" href="/vite/assets/blog_post-7eTyeLLt.webp"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDA0OSwicHVyIjoiYmxvYl9pZCJ9fQ==--a6531362dd1f3afb65f5b269e1a23113df7171b1/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Devices-amico.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzcyNywicHVyIjoiYmxvYl9pZCJ9fQ==--2d5cbbf5c3b4a73ae4b2c50632305d78f5872e4d/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programmer-rafiki.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAzMSwicHVyIjoiYmxvYl9pZCJ9fQ==--442647b9b09e64febe5646427471c53eb6f80b32/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programming-pana.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAxOSwicHVyIjoiYmxvYl9pZCJ9fQ==--84efd2b6854b7000046e9ce06e6be85d38af5ab8/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/JavaScript%20frameworks-cuate.png"/><link rel="preload" as="image" href="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzcwOSwicHVyIjoiYmxvYl9pZCJ9fQ==--03e50bbd408fef672ad099f7b2a258d80f54ad96/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Hand%20coding-bro.png"/><link rel="preload" as="image" href="/vite/assets/development-BVihs_d5.png"/><div id="app" data-page="{"component":"web/blog/posts/show","props":{"errors":{},"locale":"ru","language":"ru","httpsHost":"https://ru.hexlet.io","host":"ru.hexlet.io","colorScheme":"light","auth":{"user":{"id":null,"last_viewed_notification_id":null,"email":null,"state":null,"first_name":"","last_name":"","created_at":"2026-02-26T20:39:27.042Z","current_program":null,"current_team":null,"full_name":"","guest":true,"can_use_paid_features":false,"is_hexlet_employee":false,"sanitized_phone_number":"","can_subscribe":true,"can_renew_education":false}},"cloudflareTurnstileSiteKey":"0x4AAAAAAA15KmeFXzd2H0Xo","vkIdClientId":"51586979","yandexIdClientId":"88d071f1d3384eb4bd1deb37910235c7","formAuthToken":"oqgovUCNtmUg2j_8rO-i_JLX90tv7h959acDo7wCHBxNeeOKsvMbBZaZG2Sg4FKLUt7a4WfZ4dtIR5n37gX7cg","post":{"model_name":"BlogPost","category":{"id":4,"name":"Код","slug":"code","state":"published","created_at":"2016-08-23T13:33:44.258Z"},"creator":{"public_name":"Дмитрий Дементий","id":174372,"is_tutor":false},"tags":[{"id":633,"slug":"javascript","name":"JavaScript"}],"id":549,"title":"Идиоматический Redux: Redux Toolkit 1.0","slug":"idiomaticheskiy-redux-redux-starter-kit-1-0","state":"published","summary":"В конце октября 2019 года вышел Redux Toolkit 1.0. Один из авторов проекта проекта и автор оригинальной публикации Марк Эриксон (Mark Erikson) рассказывает, как появился этот инструмент, делится целями создания Redux Starter Kit и объясняет, как удалось выполнить задуманное. ","votes_count":8,"created_at":"2019-11-11T16:12:25.431Z","published_at":"2019-11-12T09:28:38.146Z","body":"В конце октября 2019 года вышел Redux Toolkit 1.0. Один из авторов проекта проекта и автор оригинальной публикации Марк Эриксон (Mark Erikson) рассказывает, как появился этот инструмент, делится целями создания Redux Starter Kit и объясняет, как удалось выполнить задуманное. \n\nКраткая информация о Redux Toolkit: \n\n* помогает быстро начать использовать Redux; \n* упрощает работу с типичными задачами и кодом Redux;\n* позволяет использовать лучшие практики Redux по умолчанию;\n* предлагает решения, которые уменьшают недоверие к бойлерплейтам.\n\n::programs\n\n\n## Содержание\n\n## Истоки\n\n\n### Появление Redux\n\nRedux появился летом 2015 года. Он стал логическим завершением споров и войн вокруг Flux. В течение года с момента появления он обошёл другие имплементации Flux и де-факто стал стандартным инструментом для управления состояниями в React-приложениях. \n\nRedux намеренно разрабатывался как расширяемый инструмент, и он стал именно таким инструментом. Вокруг проекта появилась целая экосистема аддонов: от [Action/Reducer Generators](https://github.com/markerikson/redux-ecosystem-links/blob/master/action-reducer-generators.md) до [Store Persistence](https://github.com/markerikson/redux-ecosystem-links/blob/master/store-persistence.md) или [десятков утилит для работы с неизменяемостью](https://github.com/markerikson/redux-ecosystem-links/blob/master/immutable-data.md). Например, поскольку в Redux нет нативных инструментов для управления асинхронным поведением и побочными эффектами, появились [десятки аддонов для работы с побочными эффектами](https://github.com/markerikson/redux-ecosystem-links/blob/master/side-effects.md). \n\nОдним словом, если вам нужно что-то сделать с помощью Redux, с высокой долей вероятности уже есть аддон, который позволяет решить вашу задачу. В противном случае вы можете создать его самостоятельно. \n\n\n### Цикл хайпа\n\nПрактически все инструменты и технологии проходят через [цикл хайпа](https://en.wikipedia.org/wiki/Hype_cycle). Когда появляется новый продукт, первые пользователи в него влюбляются и считают его серебряной пулей. Потом с новым продуктом знакомится больше людей. Они замечают ограничения и недостатки. Появляются негативные отзывы. Наконец, большинство пользователей находит лучшие способы применения технологии. Инструмент получает репутацию рабочего решения со своими преимуществами и недостатками. \n\nRedux тоже прошёл через цикл хайпа. Разработчики быстро стали ассоциировать его с React. Появилась если не догма, то устойчивое мнение об обязательном использовании Redux там, где используется React. Некоторые senior-разработчики стали говорить новичкам, что они должны использовать Redux, если используют React. \n\nВ результате многие люди стали изучать Redux без осознания контекста. Новички часто не понимали, почему появился Redux, какие проблемы решает этот инструмент. Слепое использование любого инструмента — прямой путь к большим проблемам. \n\n> **Слепое использование любого инструмента — прямой путь к большим проблемам.**\n\nНекоторые люди успешно использовали Redux и были довольны его возможностями. Другие заметили недостатки, причём не только реальные, но и придуманные. \n\n\n## Проблема сложности\n\nЕсть несколько факторов, которые в сумме делают использование Redux довольно сложным по сравнению с другими инструментами. Вот эти факторы:\n\n* Redux намеренно добавляет уровень косвенности в идею управления состояниями. Как [отмечает Дэн Абрамов](https://twitter.com/dan_abramov/status/733742952657342464), Redux не создавался в качестве самого производительного или самого простого инструмента управления мутациями. Он фокусируется на предсказуемости кода. \n* [Общепринятые практики в Redux предполагают дополнительный уровень косвенности и использование «кода, который должен быть написан»](https://blog.isquaredsoftware.com/2017/05/idiomatic-redux-tao-of-redux-part-2/): это создатели действий, константы типов действий, thunk’и, switch-выражения, неизменяемые обновления. \n* Также для Redux написано много «правил», которые, впрочем, не соблюдаются самим Redux. Например, речь идёт о неизменяемости, сериализации, избегании побочных эффектов. \n* JavaScript — мутабельный язык программирования, поэтому писать [корректный иммутабельный код](https://redux.js.org/recipes/structuring-reducers/immutable-update-patterns) на нём сложно. Это связано как с необходимостью избегать случайных мутаций, так и с большим количеством кода, который нужно написать, чтобы управлять обновлениями без изменяемости. \n* Redux всегда был инструментом с небольшим ядром, вокруг которого сформирована большая экосистема аддонов. Это позволяет пользователю выбирать настройки контейнера самостоятельно. С одной стороны, такой подход обеспечивает гибкость. С другой, даже простые задачи требуют от разработчика достаточно сложных действий, например, добавления мидлваров в асинхронную логику или настройку инструментов разработчика. Также это значит, что пользователь должен осознанно выбирать аддоны на ранних стадиях разработки, даже если приложение простое. \n* Концепции функционального программирования сложны и незнакомы большинству пользователей, особенно людям с бэкграундом в ООП.\n\nКак большинство ранних пользователей, первые пользователи Redux относились к категории продвинутых. Конфигурация, настройка и «ручная имплементация, которая позволяет контролировать все нюансы» были ключевыми преимуществами, с помощью которых инструмент привлекал новых пользователей. Тем не менее эти преимущества стали недостатками, когда в Redux пришло много новых пользователей с разным опытом и уровнем подготовки. Многие новички рассматривали паттерны и практики просто как бойлерплейт, с которым сложно работать. Поэтому они искали другие решения. \n\nВ дополнение к минимально необходимым, но сложным редьюсерам и диспетчеризации действий проблемой стала случайная сложность. Например, документация Redux рекомендует организовывать код в файлы и директории по действиям, константам, редьюсерам и контейнерам. Есть [веские причины](https://blog.isquaredsoftware.com/2017/05/idiomatic-redux-tao-of-redux-part-2/#defining-actions-action-creators-and-reducers-in-separate-files) распределять код в файлы по типам. В то же время [нет обязательных требований](https://redux.js.org/faq/code-structure#what-should-my-file-structure-look-like-how-should-i-group-my-action-creators-and-reducers-in-my-project-where-should-my-selectors-go) писать действия, редьюсеры и константы в отдельных файлах или директориях. Большинство пользователей читали документацию и выполняли рекомендации, поэтому организация кода по действиям, константам, редьюсерам и контейнером стала стандартным паттерном. \n\nПохожий пример: написание типов действий как констант ( `const ADD_TODO = \"ADD_TODO\"` ) или наличие функций, создающих действие для каждого типа, также не требуется в обязательном порядке. Но эти подходы вошли в [рекомендованные паттерны](https://redux.js.org/faq/actions#why-should-type-be-a-string-or-at-least-serializable-why-should-my-action-types-be-constants), а пользователи активно следуют этим рекомендациям. \n\nНеобходимость выбирать аддоны часто вызывала сложности у пользователей. Болевой точкой стало большое количество библиотек с побочными эффектами, в каждой из которых используется своя терминология. Уже через год после появления Redux стало понятно, что [побочные эффекты — большая проблема](https://medium.com/javascript-and-opinions/redux-side-effects-and-you-66f2e0842fc3). \n\nЭкосистема держалась [на thunk, сагах и наблюдаемых объектах](https://blog.isquaredsoftware.com/2017/09/presentation-might-need-redux-ecosystem/). Одним из самых распространённых вопросов у пользователей стал [«как мне выбрать мидлвар для работы с побочными эффектами»](https://redux.js.org/faq/actions#what-async-middleware-should-i-use-how-do-you-decide-between-thunks-sagas-observables-or-something-else). Кстати, [thunk изначально были в ядре Redux](https://github.com/reduxjs/redux/pull/63#issuecomment-110575809), но их убрали из него ради дополнительной гибкости. Очевидно, что пользователи нуждались в «официальных» инструментах для работы с побочными эффектами. \n\n\n## В поисках решения\n\nАвтор оригинальной публикации погрузился в работу над Redux в 2016 году, когда в качестве волонтёра писал [раздел FAQ](https://redux.js.org/faq) на официальном сайте проекта. Дэн Абрамов выдал автору и Тиму Дорру карт-бланш, и автор начал отвечать на вопросы пользователей на разных сайтах и форумах. \n\nК весне 2017 года автор заметил изменения в сообществе: большинство пользователей не понимали, для чего создавался Redux, а также почему существуют типичные паттерны применения этого инструмента. Автор попытался исправить это с помощью нескольких подробных публикаций: [«Дао Redux, часть 1: реализация и намерение»](https://blog.isquaredsoftware.com/2017/05/idiomatic-redux-tao-of-redux-part-1/) и [«Дао Redux, часть 2: практика и философия»](https://blog.isquaredsoftware.com/2017/05/idiomatic-redux-tao-of-redux-part-2/). Эти статьи рассказывают, для чего создан Redux и как его планировалось использовать. \n\nПримерно в то же время автор [открыл issue](https://github.com/reduxjs/redux/issues/2295) в репозитории Redux. В нём поднимались такие вопросы:\n\n* Каким должно быть идиоматическое использование Redux с условием ухода от бойлерплейта? Как мы можем реагировать на подобные жалобы?\n* Какие абстракции можно создать, чтобы упростить изучение и использование Redux без необходимости прятать этот инструмент?\n\nВ первом ответе Джастин Фалькон (Justin Falcone) предположил следующее:\n\n* Нужен официальный пакет redux-preset с redux, react-redux, redux-thunk. \n* Нужны создатели редьюсеров, например, `createReducer({[actionType]: (state, payload) => state}, initState)`. \n\nЭти предположения стали основой будущего Redux Toolkit. \n\nВ треде появилось более сотни комментариев. В итоге некоторые пользователи просто рекламировали свои библиотеки для работы с абстракциями. В середине треда Дэн Абрамов оставил [свои пожелания](https://github.com/reduxjs/redux/issues/2295#issuecomment-287960516):\n\n* Подмножество Flow/TypeScript, с которым проще работать, чем с чистым Redux. \n* Уход от выделения констант, просто сделайте строковые литералы более безопасными.\n* Сохранение независимости от React и других библиотек, но с упрощением использования существующих зависимостей, например, можно предоставить реализации `mapStateToProps`. \n* Сохранение принципов Redux: сериализуемые действия, путешествия во времени и горячая перезагрузка должны работать, журнал действий должен иметь смысл. \n* Разделение кода простым способом из коробки. \n* Поддержка использования редьюсеров с селекторами, а также создание условий, в которых они не будут казаться ужасными вместе (подумайте о парах редьюсер-селектор, которые легко писать). \n* Вместо совмещения создателей действий с редьюсерами, стоит полностью уйти от создателей действий. \n* Принять разумные значения производительности по умолчанию, чтобы мемоизация через Reselect просто работала без необходимости самостоятельно писать код. \n* Создание встроенных помощников для индексации, нормализации, коллекций. \n* Поддержка встроенного тестируемого асинхронного потока. \n* Инструмент должен быть брендированным и позиционироваться в качестве официального.\n\n\n## Начало процесса\n\nВ феврале 2018 Ник Маккерди (Nick McCurdy) открыл issue, в котором [предложил внедрить заморозку состояния на стадии разработки](https://github.com/reduxjs/redux/issues/2858). Автор оригинальной публикации написал в треде такой комментарий:\n\n«Я уже писал, что хочу создать стартовый пакет, в котором будет официально рекомендованный командой Redux и простой набор инструментов. Я посмотрел десятки вариантов, включая вариант Create React App + Redux, но пока мы ничего не одобрили. \n\nНастоящая проблема в том, что ядро Redux из коробки вообще не имеет установленных конфигураций. Мы не предполагаем, какой мидлвар будет использовать разработчик, и будет ли он вообще его использовать. Поэтому мы не можем просто добавить мидлвар в ядро, потому что в ядре из коробки нет установленных конфигураций».\n\nВ ответ Тим Дорр [открыл issue](https://github.com/reduxjs/redux/issues/2859) и написал следующее:\n\n«Похоже, для успешного начала работы с библиотекой было бы полезно сделать стартеркит с ядром Redux, в которым также должны быть несколько общих мидлваров, несколько расширителей (enhancers). \n\nВ моём представлении в стартовом пакете должна быть одна из библиотек редьюсеров, популярные мидлвары, например, thunk’и и саги, полезные инструменты разработки. Можно сделать подмножество пакетов для тех, кто использует React. \n\nНе думаю, что нужно делать что-то похожее на Create React App. Но нам нужно что-то, что из коробки даёт все инструменты, нужные для начала работы».\n\nПосле этого на Reactiflux началась продуктивная дискуссия с участием автора оригинальной публикации, Ника Маккерди и Гарри Волфа (Harry Wolff). По её итогам Гарри Волф написал, что могло бы быть в стартовом пакете Redux:\n\n* обёртка для `createStore` с хорошим набором дефолтных инструментов, например, простой способ добавления мидлваров, инструменты разработчика Redux;\n* встроенный хелпер для работы с неизменяемостью ([Immer](https://ru.hexlet.io/blog/posts/kak-immer-pokoryaet-react)?);\n* встроенный `createReducer`;\n* создатель действий, который совместим с FSA;\n* поддержка асинхронных действий, поддержка побочных эффектов — Flux, сага, что-то ещё?\n\nНа следующий день автор этой статьи [опубликовал](https://github.com/reduxjs/redux/issues/2859#issuecomment-369816441) пример реализации `configureStore` и `createReducer`. Ещё через два дня автор [опубликовал Redux Starter Kit](https://github.com/reduxjs/redux/issues/2859#issuecomment-370182309) как экспериментальный пакет. \n\n\n## Проектирование инструментов\n\n\n### Первые имплементации и нейминг\n\nРабота над Redux Toolkit продолжалась в течение нескольких следующих месяцев. Ник добавил первые юнит-тесты, интеграцию с CI, включил инструменты разработчика Redux в `configureStore`. Автор статьи добавил реэкспорт из Redux и библиотеку `selectorator`, а также улучшил настройки мидлваров. После высокой активности с марта по май 2018 наступил период затишья, который продолжался несколько месяцев. \n\nАвтор публикации ощутил выгорание летом 2018 года из-за высокой нагрузки. После обдумывания ситуации он решил сосредоточиться на обязанностях мейнтейнера проекта. Это предполагало в числе других активностей популяризацию Redux Starter Kit. \n\nВыбранное название казалось отвлечённым. Именно поэтому автор статьи опубликовал инструмент как личный пакет. В августе 2018 года он [открыл issue](https://github.com/reduxjs/redux-starter-kit/issues/36), в котором начал дискуссию о финальном названии инструмента. В числе прочего предлагалось использовать префикс @redux. \n\nДискуссия продолжалась до середины октября 2018 года. Один из участников предложил вернуться к простому варианту `redux-starter-kit`. Проблема была в том, что кто-то уже опубликовал пакет с названием `redux-toolkit` на npm. Автор статьи на всякий случай обратился к владельцу пакета с ником @shotak и спросил, не может ли он подарить имя `redux-toolkit`. К удивлению автора, человек ответил в течение нескольких часов и согласился подарить имя. После этого автор опубликовал первую официальную версию инструмента. \n\n\n### Работа над функциональностью\n\nФункции `createReducer` и `createAction` уже были полезными. Но автор статьи хотел найти более простые инструменты. \n\nКоманда [обсуждала использование хелперов](https://github.com/reduxjs/redux-starter-kit/issues/17) из пакетов типа `redux-utilities`. \n\nАвтор публикации видел десятки подобных библиотек. Одна из библиотек, которая привлекла внимание автора — [`autodux` Эрика Эллиотта](https://github.com/reduxjs/redux-starter-kit/issues/17). Она позволяет пользователям автоматически генерировать создателей действий и типы действий на основе объекта с редьюсерами и префикса с именем. \n\nШон Ванг (Shawn Wang) [оценил](https://github.com/reduxjs/redux-starter-kit/issues/17#issuecomment-414091942) `autodux` и несколько других пакетов. Вот выдержка:\n\n«Использование `autodux` даёт мощный набор создателей, селекторов и редьюсеров из коробки. Поэтому вы можете писать действия для бизнес-логики без дополнительных усилий. Мне нравится `autodux`». \n\nВ конце концов `autodux` стал источником вдохновения при создании самой полезной части Redux Toolkit: [функции `createSlice`](https://redux-starter-kit.js.org/api/createslice). \n\nЭрик Боуэр (Eric Bower) [попробовал портировать](https://github.com/reduxjs/redux-starter-kit/pull/37) `autodaz` в `createSlice`. Пулреквест застопорился. Автор статьи узнал о [библиотеке `robodux`, в которой используется Immer](https://www.reddit.com/r/javascript/comments/9fu82r/the_rise_of_immer_as_immutability_library_in_react/e5zmo3q/). Он понял, что Эрик Боуэр решил сделать собственную версию библиотеки на TypeScript. \n\nПосле обсуждения Боуэр согласился вернуть код в Redux Toolkit. Автор скомпилировал код библиотеки в чистый JavaScript и включил его в стартеркит. \n\n\n## Создание дорожной карты\n\nВ начале декабря 2018 года с помощью Docusaurus [была опубликована документация Redux Toolkit](https://github.com/reduxjs/redux-starter-kit/issues/48).\n\nАвтор статьи добавил в стартеркит дефолтные [мидлвары для определения мутаций и несериализируемых значений](https://github.com/reduxjs/redux-starter-kit/issues/48), а также [возможность трассировки стека](https://github.com/reduxjs/redux-starter-kit/pull/64).\n\nВ конце января 2019 года автор статьи [открыл issue](https://github.com/reduxjs/redux-starter-kit/issues/82), в котором предложил обсудить дорожную карту к выпуску redux Starter Kit 1.0, а также начал работать [над способностью слайсов слушать другие действия](https://blog.isquaredsoftware.com/2019/10/redux-starter-kit-1.0/). Дэнис Вашингтон (Denis Washington) прокомментировал этот пулреквест. Он предложил [обсудить целесообразность использования `createSlice`](https://github.com/reduxjs/redux-starter-kit/pull/83#pullrequestreview-194450952). \n\nВ ответ автор публикации написал большой комментарий. Вот выдержки:\n\n#### Цели\n\nСогласно README, в числе прочих источниками вдохновения для создателей Redux Toolkit являются Create-React-App и Apollo-Boost. Это пакеты, в которых есть дефолтные инструменты и абстракции над более сложными настраиваемыми инструментами. Цель существования этих пакетов — обеспечить быстрый старт при работе с библиотекой без необходимости погружаться в детали и принимать сложные решения. В то же время эти пакеты не привязывают разработчика к дефолтным инструментам и настройкам. Они очень полезны для новичков, которые не знают о всех доступных вариантах и о «правильных» выборах. Также пакеты полезны для опытных разработчиков, которым нужен простой инструмент, не требующий настройки окружения. \n\nЯ хочу, чтобы Redux Toolkit стал таким же инструментом для Redux. \n\n\n#### Быстрый старт\n\nЯ хочу, чтобы люди начинали работать с Redux как можно быстрее.\n\n\n#### Упрощение основных способов применения и уход от бойлерплейтов\n\nЯ хочу убрать острые углы:\n\n* сократить код где это возможно;\n* предложить функции, которые выполняют очевидные действия, которые пользователи привыкли делать руками;\n* посмотреть, как люди используют Redux и как они хотят его использовать, и предложить «официальное» решение на основе предпочтений пользователей. \n\nRedux Toolkit должен вести пользователя к правильным настройкам, а также автоматически защищать от распространённых ошибок или предупреждать о них. \n\n\n#### Отсутствие блокировки\n\nКогда кто-то начинает делать приложение на базе Redux Toolkit, я не хочу забирать у него возможность сделать что-то вручную в начале или в процессе работы. \n\nС другой стороны, люди должны иметь возможность добавлять Redux Toolkit в существующие приложения. В таких случаях у разработчиков должна быть возможность внедрять стартеркит постепенно без необходимости переписывать код с нуля. \n\n\n#### Стать очевидным дефолтным способом использования Redux\n\nВ долгосрочной перспективе автор видит Redux Toolkit очевидным дефолтным способом использования Redux для большинства разработчиков. В качестве примера можно посмотреть на Create React App, который стал дефолтным очевидным способом использования React. \n\n\n#### Отказ от попыток охватить все способы использования Redux, фокусировка на типичных подходах\n\nЕсть много способов использования Redux. А в ядре Redux есть минимальный набор инструментов. Именно поэтому существуют тысячи библиотек и аддонов, который охватывают разные кейсы применения Redux. Стартеркит не может и не собирается охватывать все способы применения Redux. \n\n\n#### Разумная поддержка TypeScript\n\nАвтор хочет поддерживать специалистов, которые пишут на TypeScript, как и других специалистов, которые используют Redux специфическим способом. Поэтому в Redux Toolkit должна быть хорошая типизация. В то же время автор не хочет, чтобы «лучшая типизация» стала врагом «хорошей функциональности». \n\n\n## Работа с API\n\nПараллельно с этой дискуссией Дэнис Вашингтон конвертировал базу Redux Toolkit в TypeScript. \n\nАвтор публикации потратил несколько месяцев на работу с [дорожной картой React-Redux 7](https://github.com/reduxjs/react-redux/issues/1177). Потом он работал с [дизайном хуков API](https://github.com/reduxjs/react-redux/issues/1179). \n\nОдновременно команда обсуждала разные вопросы, включая [расширение возможностей `createSlice`](https://github.com/reduxjs/redux-starter-kit/issues/91). Рассматривалась функция `combineSlices`, а также автогенерируемый селектор функциональности. \n\nК лету 2019 года автор завершил работу над React-Redux и начал писать [документацию Redux Toolkit](https://github.com/reduxjs/redux-starter-kit/pull/164). В начале сентября она была готова. \n\nТем временем Ленц Вебер (Lenz Weber) предложил несколько улучшений, включая [внедрение колбэка `prepare` в `createAction`](https://github.com/reduxjs/redux-starter-kit/pull/149), а также [улучшение читабельности определений типов](https://github.com/reduxjs/redux-starter-kit/pull/168). \n\nС этого момента автор начал «допиливать» API Redux Toolkit. Появилась [возможность кастомизации дефолтных мидлваро](https://github.com/reduxjs/redux-starter-kit/pull/192)в, а также [включение редьюсеров в `createSlice`](https://github.com/reduxjs/redux-starter-kit/pull/209). \n\nБыло ещё много идей, но [автор решил](https://github.com/reduxjs/redux-starter-kit/issues/82#issuecomment-542773046), что их можно реализовать после релиза. После этого официально вышел Redux Toolkit 1.0. \n\n\n## Будущее Redux Toolkit\n\n\n### Обзор\n\nОглядываясь на проделанную работу, можно сказать, что создателям удалось выполнить основные запланированные задачи. Redux Toolkit стал инструментом с такими характеристиками и возможностями:\n\n* официальный пакет, в котором реализованы лучшие практики;\n* значительно упрощает работу с Redux;\n* сохраняет основные принципы Redux;\n* стартеркит можно добавить в проект на любой стадии реализации;\n* полезный инструмент как для новичков, так и для опытных разработчиков;\n* поддерживает TypeScript без увеличения сложности разработки.\n\nАвтор активно рекламировал Redux Toolkit целый год. Ссылка на этот инструмент появилась на главной странице документации Redux. Также инструмент рекомендуется в разделе документации [Configuring Your Store](https://redux.js.org/recipes/configuring-your-store).\n\nСудя по [статистике загрузки пакета](https://npm-stat.com/charts.html?package=redux-starter-kit&amp;from=2018-07-01&amp;to=2019-10-23), Redux Toolkit оказался востребованным инструментом. Его устанавливают более 100 тыс. раз в месяц, и график количества загрузок показывает рост. Автор получил много положительных откликов о Redux Toolkit.\n\n::posts\n\n\n## Следующие шаги\n\nВ настоящее время [планируется](https://github.com/reduxjs/redux/issues/3313#issuecomment-450601554) масштабная работа по модернизации документации Redux. В рамках модернизации будет созданы обучающие материалы по Redux Toolkit. Также информация о стартерките будет добавлена в гайды и туториалы. Пользователи узнают о простых способах работы с Redux, например, об избегании использования констант действия или об [использовании утиного паттерна при организации логики](https://github.com/erikras/ducks-modular-redux). Возможно, в Redux Toolkit добавят [возможность определять асинхронные побочные эффекты в слайсах](https://github.com/reduxjs/redux-starter-kit/issues/76). \n\n_Адаптированный перевод статьи [Idiomatic Redux: Redux Toolkit 1.0](https://blog.isquaredsoftware.com/2019/10/redux-starter-kit-1.0/) by Mark Erikson. Мнение администрации «Хекслета» может не совпадать с мнением автора оригинальной публикации._","reading_time":10,"url":"https://ru.hexlet.io/blog/posts/idiomaticheskiy-redux-redux-starter-kit-1-0","cover_thumb_variant":null,"cover_list_variant":"/vite/assets/blog_post-7eTyeLLt.webp","cover_main_variant":"/vite/assets/blog_post-7eTyeLLt.webp","related_stacks_count":5},"relatedPosts":[{"model_name":"BlogPost","id":416,"title":"Язык программирования JavaScript: где его используют и почему он популярен","slug":"stoit-li-uchit-javascript-perspektivy-situatsiya-na-rynke-truda-mneniya-ekspertov","summary":"Рассказываем, что это за язык программирования — JavaScript, где его используют, насколько он популярен и с чего начать изучение JavaScript.","created_at":"2019-07-26T08:17:06.287Z","published_at":"2023-10-04T08:41:58.734Z","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MTcwMywicHVyIjoiYmxvYl9pZCJ9fQ==--aa69869c906a992728ba7c26886fa1b715024424/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/%D1%8F%D0%B7%D1%8B%D0%BA%D0%BF%D1%80%D0%BE%D0%B3%D1%80%D0%B0%D0%BC%D0%BC%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D1%8Fjavascript-01.png"},{"model_name":"BlogPost","id":480,"title":"Все про Node.js: зачем писать бэкенд на JavaScript и сколько за это платят","slug":"zachem-izuchat-node-js-ili-o-perspektivah-bekenda-na-javascript","summary":"Рассказываем, как устроен бэкенд на JavaScript, стоит ли изучать Node.js, когда есть PHP и Python, какое будущее у этой технологии и сколько платят программистам на Node.js.","created_at":"2019-09-27T08:33:36.150Z","published_at":"2023-07-28T13:31:55.753Z","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MTY4MywicHVyIjoiYmxvYl9pZCJ9fQ==--8e5dd13a34bb6d61f1c0db87a25e6e64d0e237fb/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/node%20_new-01.png"},{"model_name":"BlogPost","id":362,"title":"Язык программирования PHP: рейтинг, сферы применения, прогнозы экспертов","slug":"zachem-izuchat-php-reyting-perspektivy-sfery-primeneniya","summary":"Рассказываем, зачем и где учить PHP, где его применяют, сколько зарабатывают PHP-разработчики, а также приводим мнения экспертов о перспективах и популярности языка. ","created_at":"2019-06-21T09:57:49.863Z","published_at":"2023-03-24T11:43:13.709Z","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MTcxNSwicHVyIjoiYmxvYl9pZCJ9fQ==--483469e30822f45e5959af1ca8e34c28c3fe28db/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/%D1%81%D1%87%D0%B5%D0%B3%D0%BE%D0%BD%D0%B0%D1%87%D0%B0%D1%82%D1%8C%D0%B8%D0%B7%D1%83%D1%87%D0%B0%D1%82%D1%8Cphp-2-01.png"}],"category":{"id":4,"name":"Код","slug":"code","state":"published","created_at":"2016-08-23T13:33:44.258Z"},"mainStackCategory":{"id":2,"name":"Курсы по веб-разработке","slug":"web_development","short_name":"Веб-разработка","order":190,"state":"published","category_slug":"courses_web_development"},"categories":[{"id":6,"name":"Мотивация","slug":"motivation","state":"published","created_at":"2016-10-06T18:31:38.903Z"},{"id":3,"name":"Истории успеха","slug":"success","state":"published","created_at":"2016-07-30T12:57:18.308Z"},{"id":14,"name":"Дневник студента","slug":"student-diary","state":"published","created_at":"2019-02-25T13:27:09.471Z"},{"id":4,"name":"Код","slug":"code","state":"published","created_at":"2016-08-23T13:33:44.258Z"},{"id":12,"name":"Карьера","slug":"career","state":"published","created_at":"2017-07-21T15:42:21.481Z"}],"relatedLandings":[{"stack":{"id":23,"slug":"js-react-development","title":"React","audience":"for_programmers","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":350,"duration_in_months":2},"id":34,"slug":"js-react-developer","title":"React","subtitle":"Навык разрабатывать быстрые и удобные интерфейсы, открывающий доступ к интересным вакансиям в крупных компаниях","subtitle_for_lists":"Освоите React и создание быстрых интерфейсов","locale":"ru","current":true,"duration_in_months_text":"2 месяца","stack_slug":"js-react-development","price_text":"от 3 900 ₽","duration_text":"2 месяца","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDA0OSwicHVyIjoiYmxvYl9pZCJ9fQ==--a6531362dd1f3afb65f5b269e1a23113df7171b1/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Devices-amico.png"},{"stack":{"id":12,"slug":"frontend","title":"Фронтенд-разработчик","audience":"for_beginners","start_type":"weekly","pricing_model":"purchase","priority":"high","kind":"profession","state":"published","stack_state":"finished","order":20,"duration_in_months":10},"id":17,"slug":"frontend","title":"Фронтенд-разработчик","subtitle":"Изучите HTML, CSS, JavaScript и React","subtitle_for_lists":"Изучите HTML, CSS, JavaScript и React","locale":"ru","current":true,"duration_in_months_text":"10 месяцев","stack_slug":"frontend","price_text":"от 6 792 ₽","duration_text":"10 месяцев","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzcyNywicHVyIjoiYmxvYl9pZCJ9fQ==--2d5cbbf5c3b4a73ae4b2c50632305d78f5872e4d/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programmer-rafiki.png"},{"stack":{"id":21,"slug":"js-async","title":"Асинхронное программирование на JS","audience":"for_programmers","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":1650,"duration_in_months":1},"id":30,"slug":"js-async","title":"Асинхронное программирование на JS","subtitle":"Навык асинхронного программирования в JavaScript: писать быстрый код, расти в грейде и успешно проходить собеседования","subtitle_for_lists":"Навык работы с асинхронностью в JS","locale":"ru","current":true,"duration_in_months_text":"1 месяц","stack_slug":"js-async","price_text":"от 3 900 ₽","duration_text":"1 месяц","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAzMSwicHVyIjoiYmxvYl9pZCJ9fQ==--442647b9b09e64febe5646427471c53eb6f80b32/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programming-pana.png"},{"stack":{"id":29,"slug":"js-oop","title":"ООП на Javascript","audience":"for_programmers","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":4250,"duration_in_months":2},"id":46,"slug":"js-oop","title":"ООП на Javascript","subtitle":"Навык глубокого понимания архитектуры и написания чистого кода, позволяющий решать сложные задачи","subtitle_for_lists":"Изучите архитектуру и принципы чистого кода на JS","locale":"ru","current":true,"duration_in_months_text":"2 месяца","stack_slug":"js-oop","price_text":"от 3 900 ₽","duration_text":"2 месяца","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAxOSwicHVyIjoiYmxvYl9pZCJ9fQ==--84efd2b6854b7000046e9ce06e6be85d38af5ab8/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/JavaScript%20frameworks-cuate.png"},{"stack":{"id":52,"slug":"typescript","title":"Typescript","audience":"for_programmers","start_type":"anytime","pricing_model":"subscription","priority":"medium","kind":"track","state":"published","stack_state":"finished","order":200,"duration_in_months":2},"id":92,"slug":"typescript","title":"Typescript","subtitle":"Навык снижать ошибки, упрощать отладку, повышать качество кода и ускорять разработку с автодополнением и типизацией","subtitle_for_lists":"Изучите Typescript и получите навык снижать ошибки, упрощать отладку","locale":"ru","current":true,"duration_in_months_text":"2 месяца","stack_slug":"typescript","price_text":"от 3 900 ₽","duration_text":"2 месяца","cover_list_variant":"https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzcwOSwicHVyIjoiYmxvYl9pZCJ9fQ==--03e50bbd408fef672ad099f7b2a258d80f54ad96/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Hand%20coding-bro.png"}]},"url":"/blog/posts/idiomaticheskiy-redux-redux-starter-kit-1-0","version":"8f286f6358a90a7bef2263b3a6edf5a90a94fa42","encryptHistory":false,"clearHistory":false}"><style data-mantine-styles="true">:root, :host{--mantine-font-family: Arial, sans-serif;--mantine-font-family-headings: Arial, sans-serif;--mantine-heading-font-weight: normal;--mantine-radius-default: 0rem;--mantine-primary-color-filled: var(--mantine-color-indigo-filled);--mantine-primary-color-filled-hover: var(--mantine-color-indigo-filled-hover);--mantine-primary-color-light: var(--mantine-color-indigo-light);--mantine-primary-color-light-hover: var(--mantine-color-indigo-light-hover);--mantine-primary-color-light-color: var(--mantine-color-indigo-light-color);--mantine-spacing-xxl: calc(4rem * var(--mantine-scale));--mantine-font-size-xs: 12px;--mantine-font-size-sm: 14px;--mantine-font-size-md: 16px;--mantine-font-size-lg: clamp(16.0000px, calc(15.2727px + 0.2273vw), 18.0000px);--mantine-font-size-xl: clamp(16.0000px, calc(14.5455px + 0.4545vw), 20.0000px);--mantine-font-size-display-3: clamp(32.0000px, calc(26.1818px + 1.8182vw), 48.0000px);--mantine-font-size-display-2: clamp(36.0000px, calc(25.8182px + 3.1818vw), 64.0000px);--mantine-font-size-display-1: clamp(40.0000px, calc(25.4545px + 4.5455vw), 80.0000px);--mantine-font-size-h1: clamp(28.0000px, calc(23.6364px + 1.3636vw), 40.0000px);--mantine-font-size-h2: clamp(24.0000px, calc(21.0909px + 0.9091vw), 32.0000px);--mantine-font-size-h3: clamp(20.0000px, calc(17.0909px + 0.9091vw), 28.0000px);--mantine-font-size-h4: clamp(16.0000px, calc(13.0909px + 0.9091vw), 24.0000px);--mantine-font-size-h5: clamp(16.0000px, calc(14.5455px + 0.4545vw), 20.0000px);--mantine-font-size-h6: 1rem;--mantine-primary-color-0: var(--mantine-color-indigo-0);--mantine-primary-color-1: var(--mantine-color-indigo-1);--mantine-primary-color-2: var(--mantine-color-indigo-2);--mantine-primary-color-3: var(--mantine-color-indigo-3);--mantine-primary-color-4: var(--mantine-color-indigo-4);--mantine-primary-color-5: var(--mantine-color-indigo-5);--mantine-primary-color-6: var(--mantine-color-indigo-6);--mantine-primary-color-7: var(--mantine-color-indigo-7);--mantine-primary-color-8: var(--mantine-color-indigo-8);--mantine-primary-color-9: var(--mantine-color-indigo-9);--mantine-color-red-0: #ffeaea;--mantine-color-red-1: #fed4d4;--mantine-color-red-2: #f4a7a8;--mantine-color-red-3: #ec7878;--mantine-color-red-4: #e55050;--mantine-color-red-5: #e03131;--mantine-color-red-6: #e02829;--mantine-color-red-7: #c71a1c;--mantine-color-red-8: #b21218;--mantine-color-red-9: #9c0411;--mantine-color-violet-0: #fce9ff;--mantine-color-violet-1: #f1cfff;--mantine-color-violet-2: #e09bff;--mantine-color-violet-3: #d16fff;--mantine-color-violet-4: #be37fe;--mantine-color-violet-5: #b51afe;--mantine-color-violet-6: #b009ff;--mantine-color-violet-7: #9b00e4;--mantine-color-violet-8: #8a00cc;--mantine-color-violet-9: #7800b3;--mantine-color-indigo-0: #edecff;--mantine-color-indigo-1: #d6d5fe;--mantine-color-indigo-2: #aaa9f4;--mantine-color-indigo-3: #7b79eb;--mantine-color-indigo-4: #5451e4;--mantine-color-indigo-5: #3b37e0;--mantine-color-indigo-6: #2d2adf;--mantine-color-indigo-7: #1f1ec7;--mantine-color-indigo-8: #1819b2;--mantine-color-indigo-9: #0c149e;--mantine-color-cyan-0: #dffdff;--mantine-color-cyan-1: #caf5ff;--mantine-color-cyan-2: #99e8ff;--mantine-color-cyan-3: #64daff;--mantine-color-cyan-4: #3ccffe;--mantine-color-cyan-5: #24c8fe;--mantine-color-cyan-6: #00c2ff;--mantine-color-cyan-7: #00ade4;--mantine-color-cyan-8: #009acd;--mantine-color-cyan-9: #0085b5;--mantine-color-green-0: #e9fdec;--mantine-color-green-1: #d7f6dc;--mantine-color-green-2: #b0eab9;--mantine-color-green-3: #86df94;--mantine-color-green-4: #62d574;--mantine-color-green-5: #4ccf5f;--mantine-color-green-6: #3fcc54;--mantine-color-green-7: #2fb344;--mantine-color-green-8: #25a03b;--mantine-color-green-9: #138a2e;--mantine-color-yellow-0: #fff7e2;--mantine-color-yellow-1: #ffeecd;--mantine-color-yellow-2: #ffdc9c;--mantine-color-yellow-3: #ffc966;--mantine-color-yellow-4: #feb93a;--mantine-color-yellow-5: #feae1e;--mantine-color-yellow-6: #ffa90f;--mantine-color-yellow-8: #ca8200;--mantine-color-yellow-9: #af7000;--mantine-h1-font-size: clamp(28.0000px, calc(23.6364px + 1.3636vw), 40.0000px);--mantine-h1-font-weight: normal;--mantine-h2-font-size: clamp(24.0000px, calc(21.0909px + 0.9091vw), 32.0000px);--mantine-h2-font-weight: normal;--mantine-h3-font-size: clamp(20.0000px, calc(17.0909px + 0.9091vw), 28.0000px);--mantine-h3-font-weight: normal;--mantine-h4-font-size: clamp(16.0000px, calc(13.0909px + 0.9091vw), 24.0000px);--mantine-h4-font-weight: normal;--mantine-h5-font-size: clamp(16.0000px, calc(14.5455px + 0.4545vw), 20.0000px);--mantine-h5-font-weight: normal;--mantine-h6-font-size: 1rem;--mantine-h6-font-weight: normal;}
:root[data-mantine-color-scheme="dark"], :host([data-mantine-color-scheme="dark"]){--mantine-color-anchor: var(--mantine-color-text);--mantine-color-dimmed: #495057;--mantine-color-dark-filled: var(--mantine-color-dark-5);--mantine-color-dark-filled-hover: var(--mantine-color-dark-6);--mantine-color-dark-light: rgba(105, 105, 105, 0.15);--mantine-color-dark-light-hover: rgba(105, 105, 105, 0.2);--mantine-color-dark-light-color: var(--mantine-color-dark-0);--mantine-color-dark-outline: var(--mantine-color-dark-1);--mantine-color-dark-outline-hover: rgba(184, 184, 184, 0.05);--mantine-color-gray-filled: var(--mantine-color-gray-5);--mantine-color-gray-filled-hover: var(--mantine-color-gray-6);--mantine-color-gray-light: rgba(222, 226, 230, 0.15);--mantine-color-gray-light-hover: rgba(222, 226, 230, 0.2);--mantine-color-gray-light-color: var(--mantine-color-gray-0);--mantine-color-gray-outline: var(--mantine-color-gray-1);--mantine-color-gray-outline-hover: rgba(241, 243, 245, 0.05);--mantine-color-red-filled: var(--mantine-color-red-5);--mantine-color-red-filled-hover: var(--mantine-color-red-6);--mantine-color-red-light: rgba(236, 120, 120, 0.15);--mantine-color-red-light-hover: rgba(236, 120, 120, 0.2);--mantine-color-red-light-color: var(--mantine-color-red-0);--mantine-color-red-outline: var(--mantine-color-red-1);--mantine-color-red-outline-hover: rgba(254, 212, 212, 0.05);--mantine-color-pink-filled: var(--mantine-color-pink-5);--mantine-color-pink-filled-hover: var(--mantine-color-pink-6);--mantine-color-pink-light: rgba(250, 162, 193, 0.15);--mantine-color-pink-light-hover: rgba(250, 162, 193, 0.2);--mantine-color-pink-light-color: var(--mantine-color-pink-0);--mantine-color-pink-outline: var(--mantine-color-pink-1);--mantine-color-pink-outline-hover: rgba(255, 222, 235, 0.05);--mantine-color-grape-filled: var(--mantine-color-grape-5);--mantine-color-grape-filled-hover: var(--mantine-color-grape-6);--mantine-color-grape-light: rgba(229, 153, 247, 0.15);--mantine-color-grape-light-hover: rgba(229, 153, 247, 0.2);--mantine-color-grape-light-color: var(--mantine-color-grape-0);--mantine-color-grape-outline: var(--mantine-color-grape-1);--mantine-color-grape-outline-hover: rgba(243, 217, 250, 0.05);--mantine-color-violet-filled: var(--mantine-color-violet-5);--mantine-color-violet-filled-hover: var(--mantine-color-violet-6);--mantine-color-violet-light: rgba(209, 111, 255, 0.15);--mantine-color-violet-light-hover: rgba(209, 111, 255, 0.2);--mantine-color-violet-light-color: var(--mantine-color-violet-0);--mantine-color-violet-outline: var(--mantine-color-violet-1);--mantine-color-violet-outline-hover: rgba(241, 207, 255, 0.05);--mantine-color-indigo-filled: var(--mantine-color-indigo-5);--mantine-color-indigo-filled-hover: var(--mantine-color-indigo-6);--mantine-color-indigo-light: rgba(123, 121, 235, 0.15);--mantine-color-indigo-light-hover: rgba(123, 121, 235, 0.2);--mantine-color-indigo-light-color: var(--mantine-color-indigo-0);--mantine-color-indigo-outline: var(--mantine-color-indigo-1);--mantine-color-indigo-outline-hover: rgba(214, 213, 254, 0.05);--mantine-color-blue-filled: var(--mantine-color-blue-5);--mantine-color-blue-filled-hover: var(--mantine-color-blue-6);--mantine-color-blue-light: rgba(116, 192, 252, 0.15);--mantine-color-blue-light-hover: rgba(116, 192, 252, 0.2);--mantine-color-blue-light-color: var(--mantine-color-blue-0);--mantine-color-blue-outline: var(--mantine-color-blue-1);--mantine-color-blue-outline-hover: rgba(208, 235, 255, 0.05);--mantine-color-cyan-filled: var(--mantine-color-cyan-5);--mantine-color-cyan-filled-hover: var(--mantine-color-cyan-6);--mantine-color-cyan-light: rgba(100, 218, 255, 0.15);--mantine-color-cyan-light-hover: rgba(100, 218, 255, 0.2);--mantine-color-cyan-light-color: var(--mantine-color-cyan-0);--mantine-color-cyan-outline: var(--mantine-color-cyan-1);--mantine-color-cyan-outline-hover: rgba(202, 245, 255, 0.05);--mantine-color-teal-filled: var(--mantine-color-teal-5);--mantine-color-teal-filled-hover: var(--mantine-color-teal-6);--mantine-color-teal-light: rgba(99, 230, 190, 0.15);--mantine-color-teal-light-hover: rgba(99, 230, 190, 0.2);--mantine-color-teal-light-color: var(--mantine-color-teal-0);--mantine-color-teal-outline: var(--mantine-color-teal-1);--mantine-color-teal-outline-hover: rgba(195, 250, 232, 0.05);--mantine-color-green-filled: var(--mantine-color-green-5);--mantine-color-green-filled-hover: var(--mantine-color-green-6);--mantine-color-green-light: rgba(134, 223, 148, 0.15);--mantine-color-green-light-hover: rgba(134, 223, 148, 0.2);--mantine-color-green-light-color: var(--mantine-color-green-0);--mantine-color-green-outline: var(--mantine-color-green-1);--mantine-color-green-outline-hover: rgba(215, 246, 220, 0.05);--mantine-color-lime-filled: var(--mantine-color-lime-5);--mantine-color-lime-filled-hover: var(--mantine-color-lime-6);--mantine-color-lime-light: rgba(192, 235, 117, 0.15);--mantine-color-lime-light-hover: rgba(192, 235, 117, 0.2);--mantine-color-lime-light-color: var(--mantine-color-lime-0);--mantine-color-lime-outline: var(--mantine-color-lime-1);--mantine-color-lime-outline-hover: rgba(233, 250, 200, 0.05);--mantine-color-yellow-filled: var(--mantine-color-yellow-5);--mantine-color-yellow-filled-hover: var(--mantine-color-yellow-6);--mantine-color-yellow-light: rgba(255, 201, 102, 0.15);--mantine-color-yellow-light-hover: rgba(255, 201, 102, 0.2);--mantine-color-yellow-light-color: var(--mantine-color-yellow-0);--mantine-color-yellow-outline: var(--mantine-color-yellow-1);--mantine-color-yellow-outline-hover: rgba(255, 238, 205, 0.05);--mantine-color-orange-filled: var(--mantine-color-orange-5);--mantine-color-orange-filled-hover: var(--mantine-color-orange-6);--mantine-color-orange-light: rgba(255, 192, 120, 0.15);--mantine-color-orange-light-hover: rgba(255, 192, 120, 0.2);--mantine-color-orange-light-color: var(--mantine-color-orange-0);--mantine-color-orange-outline: var(--mantine-color-orange-1);--mantine-color-orange-outline-hover: rgba(255, 232, 204, 0.05);--app-cta-gradient: linear-gradient(90deg, var(--mantine-color-blue-9) 0%, var(--mantine-color-cyan-7) 100%);--app-color-surface: #2e2e2e;}
:root[data-mantine-color-scheme="light"], :host([data-mantine-color-scheme="light"]){--mantine-color-anchor: var(--mantine-color-text);--mantine-color-dimmed: #495057;--mantine-color-red-light: rgba(224, 40, 41, 0.1);--mantine-color-red-light-hover: rgba(224, 40, 41, 0.12);--mantine-color-red-outline-hover: rgba(224, 40, 41, 0.05);--mantine-color-violet-light: rgba(176, 9, 255, 0.1);--mantine-color-violet-light-hover: rgba(176, 9, 255, 0.12);--mantine-color-violet-outline-hover: rgba(176, 9, 255, 0.05);--mantine-color-indigo-light: rgba(45, 42, 223, 0.1);--mantine-color-indigo-light-hover: rgba(45, 42, 223, 0.12);--mantine-color-indigo-outline-hover: rgba(45, 42, 223, 0.05);--mantine-color-cyan-light: rgba(0, 194, 255, 0.1);--mantine-color-cyan-light-hover: rgba(0, 194, 255, 0.12);--mantine-color-cyan-outline-hover: rgba(0, 194, 255, 0.05);--mantine-color-green-light: rgba(63, 204, 84, 0.1);--mantine-color-green-light-hover: rgba(63, 204, 84, 0.12);--mantine-color-green-outline-hover: rgba(63, 204, 84, 0.05);--mantine-color-yellow-light: rgba(255, 169, 15, 0.1);--mantine-color-yellow-light-hover: rgba(255, 169, 15, 0.12);--mantine-color-yellow-outline-hover: rgba(255, 169, 15, 0.05);--app-color-surface: #f1f3f5;--app-cta-gradient: linear-gradient(90deg, var(--mantine-color-blue-filled) 0%, var(--mantine-color-cyan-5) 100%);}</style><style data-mantine-styles="classes">@media (max-width: 35.99375em) {.mantine-visible-from-xs {display: none !important;}}@media (min-width: 36em) {.mantine-hidden-from-xs {display: none !important;}}@media (max-width: 47.99375em) {.mantine-visible-from-sm {display: none !important;}}@media (min-width: 48em) {.mantine-hidden-from-sm {display: none !important;}}@media (max-width: 61.99375em) {.mantine-visible-from-md {display: none !important;}}@media (min-width: 62em) {.mantine-hidden-from-md {display: none !important;}}@media (max-width: 74.99375em) {.mantine-visible-from-lg {display: none !important;}}@media (min-width: 75em) {.mantine-hidden-from-lg {display: none !important;}}@media (max-width: 87.99375em) {.mantine-visible-from-xl {display: none !important;}}@media (min-width: 88em) {.mantine-hidden-from-xl {display: none !important;}}</style><script type="application/ld+json">{"@context":"https://schema.org","@type":"Article","author":"Дмитрий Дементий","name":"Идиоматический Redux: Redux Toolkit 1.0","datePublished":"2019-11-12T09:28:38.146Z","headline":"В конце октября 2019 года вышел Redux Toolkit 1.0. Один из авторов проекта проекта и автор оригинальной публикации Марк Эриксон (Mark Erikson) рассказывает, как появился этот инструмент, делится целями создания Redux Starter Kit и объясняет, как удалось выполнить задуманное. ","image":"/vite/assets/blog_post-7eTyeLLt.webp","interactionStatistic":[{"@type":"InteractionCounter","interactionType":{"@type":"LikeAction"},"userInteractionCount":8}]}</script><div style="--container-size:var(--container-size-lg);margin-top:var(--mantine-spacing-xl);height:100%" class="m_7485cace mantine-Container-root" data-size="lg" data-strategy="block"><script type="application/ld+json">{"@context":"https://schema.org","@type":"BreadcrumbList","itemListElement":[{"position":1,"@type":"ListItem","item":{"@id":"/blog","name":"Блог Хекслета"}},{"position":2,"@type":"ListItem","item":{"@id":"/blog/categories/code","name":"Код"}},{"position":3,"@type":"ListItem","item":{"@id":"/blog/posts/idiomaticheskiy-redux-redux-starter-kit-1-0","name":"Идиоматический Redux: Redux Toolkit 1.0"}}]}</script><div style="margin-bottom:var(--mantine-spacing-xs)" class="m_8b3717df mantine-Breadcrumbs-root"><a style="--text-fz:var(--mantine-font-size-sm);--text-lh:var(--mantine-line-height-sm);white-space:normal;color:inherit" class="mantine-focus-auto m_849cf0da m_f678d540 mantine-Breadcrumbs-breadcrumb m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-size="sm" data-underline="hover" href="/"><div style="color:inherit" class="m_4451eb3a mantine-Center-root"><svg xmlns="http://www.w3.org/2000/svg" width="15" height="15" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-home-link "><path d="M20.085 11.085l-8.085 -8.085l-9 9h2v7a2 2 0 0 0 2 2h4.5"></path><path d="M9 21v-6a2 2 0 0 1 2 -2h2a2 2 0 0 1 1.807 1.143"></path><path d="M20 21a1 1 0 1 0 2 0a1 1 0 1 0 -2 0"></path><path d="M20 16a1 1 0 1 0 2 0a1 1 0 1 0 -2 0"></path><path d="M15 19a1 1 0 1 0 2 0a1 1 0 1 0 -2 0"></path><path d="M21 16l-5 3l5 2"></path></svg></div></a><div class="m_3b8f2208 mantine-Breadcrumbs-separator">/</div><a style="--text-fz:var(--mantine-font-size-sm);--text-lh:var(--mantine-line-height-sm);white-space:normal;color:inherit" class="mantine-focus-auto m_849cf0da m_f678d540 mantine-Breadcrumbs-breadcrumb m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-size="sm" data-underline="hover" href="/blog">Блог Хекслета</a><div class="m_3b8f2208 mantine-Breadcrumbs-separator">/</div><a style="--text-fz:var(--mantine-font-size-sm);--text-lh:var(--mantine-line-height-sm);white-space:normal;color:inherit" class="mantine-focus-auto m_849cf0da m_f678d540 mantine-Breadcrumbs-breadcrumb m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-size="sm" data-underline="hover" href="/blog/categories/code">Код</a><div class="m_3b8f2208 mantine-Breadcrumbs-separator">/</div><p style="--text-fz:var(--mantine-font-size-sm);--text-lh:var(--mantine-line-height-sm);white-space:normal;color:var(--mantine-color-dimmed)" class="mantine-focus-auto m_f678d540 mantine-Breadcrumbs-breadcrumb m_b6d8b162 mantine-Text-root" data-size="sm">Идиоматический Redux: Redux Toolkit 1.0</p></div><style data-mantine-styles="inline">.__m__-_R_eub_{margin-bottom:var(--mantine-spacing-xs);}@media(min-width: 36em){.__m__-_R_eub_{margin-bottom:var(--mantine-spacing-xs);}}</style><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap" class="m_4081bf90 mantine-Group-root __m__-_R_eub_"><style data-mantine-styles="inline">.__m__-_R_deub_{width:100%;}@media(min-width: 36em){.__m__-_R_deub_{width:70%;}}@media(min-width: 75em){.__m__-_R_deub_{width:75%;}}</style><div class="__m__-_R_deub_"><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><h1 style="--title-fw:var(--mantine-h1-font-weight);--title-lh:var(--mantine-h1-line-height);--title-fz:var(--mantine-h1-font-size)" class="m_8a5d1357 mantine-Title-root" data-order="1">Идиоматический Redux: Redux Toolkit 1.0</h1></div></div></div><div style="position:absolute;top:calc(18.75rem * var(--mantine-scale))" class=""></div><style data-mantine-styles="inline">.__m__-_R_2iub_{--grid-gutter:var(--mantine-spacing-xl);}</style><div class="m_410352e9 mantine-Grid-root __m__-_R_2iub_"><div class="m_dee7bd2f mantine-Grid-inner"><style data-mantine-styles="inline">.__m__-_R_dmiub_{--col-flex-grow:auto;--col-flex-basis:100%;--col-max-width:100%;}@media(min-width: 48em){.__m__-_R_dmiub_{--col-flex-grow:auto;--col-flex-basis:83.33333333333334%;--col-max-width:83.33333333333334%;}}@media(min-width: 62em){.__m__-_R_dmiub_{--col-flex-grow:auto;--col-flex-basis:66.66666666666667%;--col-max-width:66.66666666666667%;}}</style><div class="m_96bdd299 mantine-Grid-col __m__-_R_dmiub_"><div style="--stack-gap:var(--mantine-spacing-md);--stack-align:stretch;--stack-justify:flex-start;margin-bottom:var(--mantine-spacing-xl)" class="m_6d731127 mantine-Stack-root"><div class=""><div style="--group-gap:var(--mantine-spacing-xs);--group-align:center;--group-justify:flex-start;--group-wrap:wrap;margin-bottom:var(--mantine-spacing-xl)" class="m_4081bf90 mantine-Group-root"><button style="--badge-height:var(--badge-height-sm);--badge-padding-x:var(--badge-padding-x-sm);--badge-fz:var(--badge-fz-sm);--badge-bg:var(--mantine-color-default);--badge-color:var(--mantine-color-default-color);--badge-bd:calc(0.0625rem * var(--mantine-scale)) solid var(--mantine-color-default-border);cursor:pointer;color:inherit" class="m_347db0ec mantine-Badge-root" data-variant="default" data-size="sm" type="button" aria-label="JavaScript"><span class="m_5add502a mantine-Badge-label">JavaScript</span></button></div><div style="--group-gap:calc(0.625rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:wrap;margin-bottom:var(--mantine-spacing-sm);color:var(--mantine-color-gray-text)" class="m_4081bf90 mantine-Group-root"><div style="--group-gap:calc(0.1875rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:wrap;margin-inline-end:var(--mantine-spacing-lg)" class="m_4081bf90 mantine-Group-root">12 ноября 2019 г.</div><div style="--group-gap:calc(0.1875rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:wrap" class="m_4081bf90 mantine-Group-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-clock "><path d="M3 12a9 9 0 1 0 18 0a9 9 0 0 0 -18 0"></path><path d="M12 7v5l3 3"></path></svg></div>10 минут</div><div style="--group-gap:calc(0.1875rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:wrap" class="m_4081bf90 mantine-Group-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-thumb-up "><path d="M7 11v8a1 1 0 0 1 -1 1h-2a1 1 0 0 1 -1 -1v-7a1 1 0 0 1 1 -1h3a4 4 0 0 0 4 -4v-1a2 2 0 0 1 4 0v5h3a2 2 0 0 1 2 2l-1 5a2 3 0 0 1 -2 2h-7a3 3 0 0 1 -3 -3"></path></svg></div>8</div></div><div style="--ar-ratio:2" class="m_71ac47fc mantine-AspectRatio-root"><img style="--image-radius:var(--mantine-radius-md);--image-object-fit:cover;width:100%;height:100%" class="m_9e117634 mantine-Image-root" src="/vite/assets/blog_post-7eTyeLLt.webp" alt="Идиоматический Redux: Redux Toolkit 1.0"/></div></div><div role="link" tabindex="0" style="cursor:pointer"><button style="display:block;width:100%" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Присоединяйтесь к нашему Telegram-сообществу"><div style="background-color:light-dark(var(--mantine-color-gray-1), var(--mantine-color-dark-6))" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root"><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:flex-start;--group-wrap:wrap" class="m_4081bf90 mantine-Group-root"><div style="--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;margin-inline-end:auto;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-brand-telegram "><path d="M15 10l-4 4l6 6l4 -16l-18 7l4 2l2 6l3 -4"></path></svg></div>Присоединяйтесь к нашему Telegram-сообществу</div></div></button></div><div style="margin-bottom:var(--mantine-spacing-xl)" class="m_d08caa0 mantine-Typography-root"><p>В конце октября 2019 года вышел Redux Toolkit 1.0. Один из авторов проекта проекта и автор оригинальной публикации Марк Эриксон (Mark Erikson) рассказывает, как появился этот инструмент, делится целями создания Redux Starter Kit и объясняет, как удалось выполнить задуманное.</p>
<p>Краткая информация о Redux Toolkit:</p>
<ul>
<li>помогает быстро начать использовать Redux;</li>
<li>упрощает работу с типичными задачами и кодом Redux;</li>
<li>позволяет использовать лучшие практики Redux по умолчанию;</li>
<li>предлагает решения, которые уменьшают недоверие к бойлерплейтам.</li>
</ul>
<style data-mantine-styles="inline">.__m__-_R_7derddmiub_{--carousel-slide-gap:var(--mantine-spacing-xs);--carousel-slide-size:70%;}@media(min-width: 36em){.__m__-_R_7derddmiub_{--carousel-slide-gap:var(--mantine-spacing-xl);--carousel-slide-size:50%;}}</style><div style="--carousel-control-size:calc(2.5rem * var(--mantine-scale));--carousel-controls-offset:var(--mantine-spacing-sm);margin-bottom:var(--mantine-spacing-lg);padding-block:var(--mantine-spacing-sm);background:var(--app-color-surface)" class="m_17884d0f mantine-Carousel-root responsiveClassName" data-orientation="horizontal" data-include-gap-in-size="true"><div class="m_39bc3463 mantine-Carousel-controls" data-orientation="horizontal"><button class="mantine-focus-auto m_64f58e10 mantine-Carousel-control m_87cf2631 mantine-UnstyledButton-root" type="button" data-inactive="true" data-type="previous" tabindex="-1"><svg viewBox="0 0 15 15" fill="none" xmlns="http://www.w3.org/2000/svg" style="transform:rotate(90deg);width:calc(1rem * var(--mantine-scale));height:calc(1rem * var(--mantine-scale));display:block"><path d="M3.13523 6.15803C3.3241 5.95657 3.64052 5.94637 3.84197 6.13523L7.5 9.56464L11.158 6.13523C11.3595 5.94637 11.6759 5.95657 11.8648 6.15803C12.0536 6.35949 12.0434 6.67591 11.842 6.86477L7.84197 10.6148C7.64964 10.7951 7.35036 10.7951 7.15803 10.6148L3.15803 6.86477C2.95657 6.67591 2.94637 6.35949 3.13523 6.15803Z" fill="currentColor" fill-rule="evenodd" clip-rule="evenodd"></path></svg></button><button class="mantine-focus-auto m_64f58e10 mantine-Carousel-control m_87cf2631 mantine-UnstyledButton-root" type="button" data-inactive="true" data-type="next" tabindex="-1"><svg viewBox="0 0 15 15" fill="none" xmlns="http://www.w3.org/2000/svg" style="transform:rotate(-90deg);width:calc(1rem * var(--mantine-scale));height:calc(1rem * var(--mantine-scale));display:block"><path d="M3.13523 6.15803C3.3241 5.95657 3.64052 5.94637 3.84197 6.13523L7.5 9.56464L11.158 6.13523C11.3595 5.94637 11.6759 5.95657 11.8648 6.15803C12.0536 6.35949 12.0434 6.67591 11.842 6.86477L7.84197 10.6148C7.64964 10.7951 7.35036 10.7951 7.15803 10.6148L3.15803 6.86477C2.95657 6.67591 2.94637 6.35949 3.13523 6.15803Z" fill="currentColor" fill-rule="evenodd" clip-rule="evenodd"></path></svg></button></div><div class="m_a2dae653 mantine-Carousel-viewport" data-type="media"><div class="m_fcd81474 mantine-Carousel-container __m__-_R_7derddmiub_" data-orientation="horizontal"><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/js-react-developer?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">2 месяца</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Для продвинутых</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">React</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Освоите React и создание быстрых интерфейсов</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDA0OSwicHVyIjoiYmxvYl9pZCJ9fQ==--a6531362dd1f3afb65f5b269e1a23113df7171b1/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Devices-amico.png" alt="React" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/frontend?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">10 месяцев</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">С нуля</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Фронтенд-разработчик</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Изучите HTML, CSS, JavaScript и React</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzcyNywicHVyIjoiYmxvYl9pZCJ9fQ==--2d5cbbf5c3b4a73ae4b2c50632305d78f5872e4d/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programmer-rafiki.png" alt="Фронтенд-разработчик" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 6 792 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/js-async?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">1 месяц</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Для продвинутых</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Асинхронное программирование на JS</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Навык работы с асинхронностью в JS</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAzMSwicHVyIjoiYmxvYl9pZCJ9fQ==--442647b9b09e64febe5646427471c53eb6f80b32/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Programming-pana.png" alt="Асинхронное программирование на JS" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/js-oop?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">2 месяца</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Для продвинутых</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">ООП на Javascript</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Изучите архитектуру и принципы чистого кода на JS</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6NDAxOSwicHVyIjoiYmxvYl9pZCJ9fQ==--84efd2b6854b7000046e9ce06e6be85d38af5ab8/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/JavaScript%20frameworks-cuate.png" alt="ООП на Javascript" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/programs/typescript?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card" target="_blank"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="--group-gap:calc(0.25rem * var(--mantine-scale));--group-align:center;--group-justify:flex-start;--group-wrap:nowrap" class="m_4081bf90 mantine-Group-root"><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">2 месяца</span><span class="mantine-focus-auto m_b6d8b162 mantine-Text-root">·</span><span style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Для продвинутых</span></div><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h5);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Typescript</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Изучите Typescript и получите навык снижать ошибки, упрощать отладку</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MzcwOSwicHVyIjoiYmxvYl9pZCJ9fQ==--03e50bbd408fef672ad099f7b2a258d80f54ad96/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX3RvX2xpbWl0IjpbNDAwLDQwMF0sInNhdmVyIjp7InF1YWxpdHkiOjg1fX0sInB1ciI6InZhcmlhdGlvbiJ9fQ==--5b6f46dacd1af664f27558553a58076185091823/Hand%20coding-bro.png" alt="Typescript" loading="eager"/></div><div style="--group-gap:var(--mantine-spacing-md);--group-align:end;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-xs)" class="m_4081bf90 mantine-Group-root"><p style="font-size:var(--mantine-font-size-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">от 3 900 ₽</p><p style="font-size:var(--mantine-font-size-sm)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/courses?promo_name=programs_list&promo_position=blog_post&promo_creative=catalog_card&promo_type=card"><div style="height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><h2 style="--title-fw:var(--mantine-h2-font-weight);--title-lh:var(--mantine-h2-line-height);--title-fz:var(--mantine-h2-font-size);margin-bottom:var(--mantine-spacing-md);font-size:var(--mantine-font-size-h3)" class="m_8a5d1357 mantine-Title-root" data-order="2" data-responsive="true">Каталог</h2><p style="margin-bottom:auto" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Полный список доступных курсов по разным направлениям</p><div style="margin-top:auto" class=""><div class="m_4451eb3a mantine-Center-root"><img style="opacity:0.8;width:70%" class="m_9e117634 mantine-Image-root mantine-visible-from-xs" src="/vite/assets/development-BVihs_d5.png" alt="Orientation"/></div></div></div></a></div></div></div></div></div>
<h2 id="heading-2-1">Содержание</h2>
<ul>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-2">Истоки</a></li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-5">Проблема сложности</a></li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-6">В поисках решения</a></li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-7">Начало процесса</a></li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-8">Проектирование инструментов</a></li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-11">Создание дорожной карты</a></li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-19">Работа с API</a></li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-20">Будущее Redux Toolkit</a></li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="#heading-2-22">Следующие шаги</a></li>
</ul>
<h2 id="heading-2-2">Истоки</h2>
<h3 id="heading-3-3">Появление Redux</h3>
<p>Redux появился летом 2015 года. Он стал логическим завершением споров и войн вокруг Flux. В течение года с момента появления он обошёл другие имплементации Flux и де-факто стал стандартным инструментом для управления состояниями в React-приложениях.</p>
<p>Redux намеренно разрабатывался как расширяемый инструмент, и он стал именно таким инструментом. Вокруг проекта появилась целая экосистема аддонов: от <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/markerikson/redux-ecosystem-links/blob/master/action-reducer-generators.md" rel="noopener noreferrer" target="_blank">Action/Reducer Generators</a> до <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/markerikson/redux-ecosystem-links/blob/master/store-persistence.md" rel="noopener noreferrer" target="_blank">Store Persistence</a> или <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/markerikson/redux-ecosystem-links/blob/master/immutable-data.md" rel="noopener noreferrer" target="_blank">десятков утилит для работы с неизменяемостью</a>. Например, поскольку в Redux нет нативных инструментов для управления асинхронным поведением и побочными эффектами, появились <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/markerikson/redux-ecosystem-links/blob/master/side-effects.md" rel="noopener noreferrer" target="_blank">десятки аддонов для работы с побочными эффектами</a>.</p>
<p>Одним словом, если вам нужно что-то сделать с помощью Redux, с высокой долей вероятности уже есть аддон, который позволяет решить вашу задачу. В противном случае вы можете создать его самостоятельно.</p>
<h3 id="heading-3-4">Цикл хайпа</h3>
<p>Практически все инструменты и технологии проходят через <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://en.wikipedia.org/wiki/Hype_cycle" rel="noopener noreferrer" target="_blank">цикл хайпа</a>. Когда появляется новый продукт, первые пользователи в него влюбляются и считают его серебряной пулей. Потом с новым продуктом знакомится больше людей. Они замечают ограничения и недостатки. Появляются негативные отзывы. Наконец, большинство пользователей находит лучшие способы применения технологии. Инструмент получает репутацию рабочего решения со своими преимуществами и недостатками.</p>
<p>Redux тоже прошёл через цикл хайпа. Разработчики быстро стали ассоциировать его с React. Появилась если не догма, то устойчивое мнение об обязательном использовании Redux там, где используется React. Некоторые senior-разработчики стали говорить новичкам, что они должны использовать Redux, если используют React.</p>
<p>В результате многие люди стали изучать Redux без осознания контекста. Новички часто не понимали, почему появился Redux, какие проблемы решает этот инструмент. Слепое использование любого инструмента — прямой путь к большим проблемам.</p>
<blockquote>
<p><strong>Слепое использование любого инструмента — прямой путь к большим проблемам.</strong></p>
</blockquote>
<p>Некоторые люди успешно использовали Redux и были довольны его возможностями. Другие заметили недостатки, причём не только реальные, но и придуманные.</p>
<h2 id="heading-2-5">Проблема сложности</h2>
<p>Есть несколько факторов, которые в сумме делают использование Redux довольно сложным по сравнению с другими инструментами. Вот эти факторы:</p>
<ul>
<li>Redux намеренно добавляет уровень косвенности в идею управления состояниями. Как <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://twitter.com/dan_abramov/status/733742952657342464" rel="noopener noreferrer" target="_blank">отмечает Дэн Абрамов</a>, Redux не создавался в качестве самого производительного или самого простого инструмента управления мутациями. Он фокусируется на предсказуемости кода.</li>
<li><a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://blog.isquaredsoftware.com/2017/05/idiomatic-redux-tao-of-redux-part-2/" rel="noopener noreferrer" target="_blank">Общепринятые практики в Redux предполагают дополнительный уровень косвенности и использование «кода, который должен быть написан»</a>: это создатели действий, константы типов действий, thunk’и, switch-выражения, неизменяемые обновления.</li>
<li>Также для Redux написано много «правил», которые, впрочем, не соблюдаются самим Redux. Например, речь идёт о неизменяемости, сериализации, избегании побочных эффектов.</li>
<li>JavaScript — мутабельный язык программирования, поэтому писать <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://redux.js.org/recipes/structuring-reducers/immutable-update-patterns" rel="noopener noreferrer" target="_blank">корректный иммутабельный код</a> на нём сложно. Это связано как с необходимостью избегать случайных мутаций, так и с большим количеством кода, который нужно написать, чтобы управлять обновлениями без изменяемости.</li>
<li>Redux всегда был инструментом с небольшим ядром, вокруг которого сформирована большая экосистема аддонов. Это позволяет пользователю выбирать настройки контейнера самостоятельно. С одной стороны, такой подход обеспечивает гибкость. С другой, даже простые задачи требуют от разработчика достаточно сложных действий, например, добавления мидлваров в асинхронную логику или настройку инструментов разработчика. Также это значит, что пользователь должен осознанно выбирать аддоны на ранних стадиях разработки, даже если приложение простое.</li>
<li>Концепции функционального программирования сложны и незнакомы большинству пользователей, особенно людям с бэкграундом в ООП.</li>
</ul>
<p>Как большинство ранних пользователей, первые пользователи Redux относились к категории продвинутых. Конфигурация, настройка и «ручная имплементация, которая позволяет контролировать все нюансы» были ключевыми преимуществами, с помощью которых инструмент привлекал новых пользователей. Тем не менее эти преимущества стали недостатками, когда в Redux пришло много новых пользователей с разным опытом и уровнем подготовки. Многие новички рассматривали паттерны и практики просто как бойлерплейт, с которым сложно работать. Поэтому они искали другие решения.</p>
<p>В дополнение к минимально необходимым, но сложным редьюсерам и диспетчеризации действий проблемой стала случайная сложность. Например, документация Redux рекомендует организовывать код в файлы и директории по действиям, константам, редьюсерам и контейнерам. Есть <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://blog.isquaredsoftware.com/2017/05/idiomatic-redux-tao-of-redux-part-2/#defining-actions-action-creators-and-reducers-in-separate-files" rel="noopener noreferrer" target="_blank">веские причины</a> распределять код в файлы по типам. В то же время <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://redux.js.org/faq/code-structure#what-should-my-file-structure-look-like-how-should-i-group-my-action-creators-and-reducers-in-my-project-where-should-my-selectors-go" rel="noopener noreferrer" target="_blank">нет обязательных требований</a> писать действия, редьюсеры и константы в отдельных файлах или директориях. Большинство пользователей читали документацию и выполняли рекомендации, поэтому организация кода по действиям, константам, редьюсерам и контейнером стала стандартным паттерном.</p>
<p>Похожий пример: написание типов действий как констант ( <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">const ADD_TODO = "ADD_TODO"</code> ) или наличие функций, создающих действие для каждого типа, также не требуется в обязательном порядке. Но эти подходы вошли в <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://redux.js.org/faq/actions#why-should-type-be-a-string-or-at-least-serializable-why-should-my-action-types-be-constants" rel="noopener noreferrer" target="_blank">рекомендованные паттерны</a>, а пользователи активно следуют этим рекомендациям.</p>
<p>Необходимость выбирать аддоны часто вызывала сложности у пользователей. Болевой точкой стало большое количество библиотек с побочными эффектами, в каждой из которых используется своя терминология. Уже через год после появления Redux стало понятно, что <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://medium.com/javascript-and-opinions/redux-side-effects-and-you-66f2e0842fc3" rel="noopener noreferrer" target="_blank">побочные эффекты — большая проблема</a>.</p>
<p>Экосистема держалась <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://blog.isquaredsoftware.com/2017/09/presentation-might-need-redux-ecosystem/" rel="noopener noreferrer" target="_blank">на thunk, сагах и наблюдаемых объектах</a>. Одним из самых распространённых вопросов у пользователей стал <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://redux.js.org/faq/actions#what-async-middleware-should-i-use-how-do-you-decide-between-thunks-sagas-observables-or-something-else" rel="noopener noreferrer" target="_blank">«как мне выбрать мидлвар для работы с побочными эффектами»</a>. Кстати, <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux/pull/63#issuecomment-110575809" rel="noopener noreferrer" target="_blank">thunk изначально были в ядре Redux</a>, но их убрали из него ради дополнительной гибкости. Очевидно, что пользователи нуждались в «официальных» инструментах для работы с побочными эффектами.</p>
<h2 id="heading-2-6">В поисках решения</h2>
<p>Автор оригинальной публикации погрузился в работу над Redux в 2016 году, когда в качестве волонтёра писал <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://redux.js.org/faq" rel="noopener noreferrer" target="_blank">раздел FAQ</a> на официальном сайте проекта. Дэн Абрамов выдал автору и Тиму Дорру карт-бланш, и автор начал отвечать на вопросы пользователей на разных сайтах и форумах.</p>
<p>К весне 2017 года автор заметил изменения в сообществе: большинство пользователей не понимали, для чего создавался Redux, а также почему существуют типичные паттерны применения этого инструмента. Автор попытался исправить это с помощью нескольких подробных публикаций: <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://blog.isquaredsoftware.com/2017/05/idiomatic-redux-tao-of-redux-part-1/" rel="noopener noreferrer" target="_blank">«Дао Redux, часть 1: реализация и намерение»</a> и <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://blog.isquaredsoftware.com/2017/05/idiomatic-redux-tao-of-redux-part-2/" rel="noopener noreferrer" target="_blank">«Дао Redux, часть 2: практика и философия»</a>. Эти статьи рассказывают, для чего создан Redux и как его планировалось использовать.</p>
<p>Примерно в то же время автор <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux/issues/2295" rel="noopener noreferrer" target="_blank">открыл issue</a> в репозитории Redux. В нём поднимались такие вопросы:</p>
<ul>
<li>Каким должно быть идиоматическое использование Redux с условием ухода от бойлерплейта? Как мы можем реагировать на подобные жалобы?</li>
<li>Какие абстракции можно создать, чтобы упростить изучение и использование Redux без необходимости прятать этот инструмент?</li>
</ul>
<p>В первом ответе Джастин Фалькон (Justin Falcone) предположил следующее:</p>
<ul>
<li>Нужен официальный пакет redux-preset с redux, react-redux, redux-thunk.</li>
<li>Нужны создатели редьюсеров, например, <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createReducer({[actionType]: (state, payload) => state}, initState)</code>.</li>
</ul>
<p>Эти предположения стали основой будущего Redux Toolkit.</p>
<p>В треде появилось более сотни комментариев. В итоге некоторые пользователи просто рекламировали свои библиотеки для работы с абстракциями. В середине треда Дэн Абрамов оставил <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux/issues/2295#issuecomment-287960516" rel="noopener noreferrer" target="_blank">свои пожелания</a>:</p>
<ul>
<li>Подмножество Flow/TypeScript, с которым проще работать, чем с чистым Redux.</li>
<li>Уход от выделения констант, просто сделайте строковые литералы более безопасными.</li>
<li>Сохранение независимости от React и других библиотек, но с упрощением использования существующих зависимостей, например, можно предоставить реализации <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">mapStateToProps</code>.</li>
<li>Сохранение принципов Redux: сериализуемые действия, путешествия во времени и горячая перезагрузка должны работать, журнал действий должен иметь смысл.</li>
<li>Разделение кода простым способом из коробки.</li>
<li>Поддержка использования редьюсеров с селекторами, а также создание условий, в которых они не будут казаться ужасными вместе (подумайте о парах редьюсер-селектор, которые легко писать).</li>
<li>Вместо совмещения создателей действий с редьюсерами, стоит полностью уйти от создателей действий.</li>
<li>Принять разумные значения производительности по умолчанию, чтобы мемоизация через Reselect просто работала без необходимости самостоятельно писать код.</li>
<li>Создание встроенных помощников для индексации, нормализации, коллекций.</li>
<li>Поддержка встроенного тестируемого асинхронного потока.</li>
<li>Инструмент должен быть брендированным и позиционироваться в качестве официального.</li>
</ul>
<h2 id="heading-2-7">Начало процесса</h2>
<p>В феврале 2018 Ник Маккерди (Nick McCurdy) открыл issue, в котором <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux/issues/2858" rel="noopener noreferrer" target="_blank">предложил внедрить заморозку состояния на стадии разработки</a>. Автор оригинальной публикации написал в треде такой комментарий:</p>
<p>«Я уже писал, что хочу создать стартовый пакет, в котором будет официально рекомендованный командой Redux и простой набор инструментов. Я посмотрел десятки вариантов, включая вариант Create React App + Redux, но пока мы ничего не одобрили.</p>
<p>Настоящая проблема в том, что ядро Redux из коробки вообще не имеет установленных конфигураций. Мы не предполагаем, какой мидлвар будет использовать разработчик, и будет ли он вообще его использовать. Поэтому мы не можем просто добавить мидлвар в ядро, потому что в ядре из коробки нет установленных конфигураций».</p>
<p>В ответ Тим Дорр <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux/issues/2859" rel="noopener noreferrer" target="_blank">открыл issue</a> и написал следующее:</p>
<p>«Похоже, для успешного начала работы с библиотекой было бы полезно сделать стартеркит с ядром Redux, в которым также должны быть несколько общих мидлваров, несколько расширителей (enhancers).</p>
<p>В моём представлении в стартовом пакете должна быть одна из библиотек редьюсеров, популярные мидлвары, например, thunk’и и саги, полезные инструменты разработки. Можно сделать подмножество пакетов для тех, кто использует React.</p>
<p>Не думаю, что нужно делать что-то похожее на Create React App. Но нам нужно что-то, что из коробки даёт все инструменты, нужные для начала работы».</p>
<p>После этого на Reactiflux началась продуктивная дискуссия с участием автора оригинальной публикации, Ника Маккерди и Гарри Волфа (Harry Wolff). По её итогам Гарри Волф написал, что могло бы быть в стартовом пакете Redux:</p>
<ul>
<li>обёртка для <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createStore</code> с хорошим набором дефолтных инструментов, например, простой способ добавления мидлваров, инструменты разработчика Redux;</li>
<li>встроенный хелпер для работы с неизменяемостью (<a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://ru.hexlet.io/blog/posts/kak-immer-pokoryaet-react" rel="noopener noreferrer" target="_blank">Immer</a>?);</li>
<li>встроенный <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createReducer</code>;</li>
<li>создатель действий, который совместим с FSA;</li>
<li>поддержка асинхронных действий, поддержка побочных эффектов — Flux, сага, что-то ещё?</li>
</ul>
<p>На следующий день автор этой статьи <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux/issues/2859#issuecomment-369816441" rel="noopener noreferrer" target="_blank">опубликовал</a> пример реализации <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">configureStore</code> и <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createReducer</code>. Ещё через два дня автор <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux/issues/2859#issuecomment-370182309" rel="noopener noreferrer" target="_blank">опубликовал Redux Starter Kit</a> как экспериментальный пакет.</p>
<h2 id="heading-2-8">Проектирование инструментов</h2>
<h3 id="heading-3-9">Первые имплементации и нейминг</h3>
<p>Работа над Redux Toolkit продолжалась в течение нескольких следующих месяцев. Ник добавил первые юнит-тесты, интеграцию с CI, включил инструменты разработчика Redux в <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">configureStore</code>. Автор статьи добавил реэкспорт из Redux и библиотеку <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">selectorator</code>, а также улучшил настройки мидлваров. После высокой активности с марта по май 2018 наступил период затишья, который продолжался несколько месяцев.</p>
<p>Автор публикации ощутил выгорание летом 2018 года из-за высокой нагрузки. После обдумывания ситуации он решил сосредоточиться на обязанностях мейнтейнера проекта. Это предполагало в числе других активностей популяризацию Redux Starter Kit.</p>
<p>Выбранное название казалось отвлечённым. Именно поэтому автор статьи опубликовал инструмент как личный пакет. В августе 2018 года он <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/36" rel="noopener noreferrer" target="_blank">открыл issue</a>, в котором начал дискуссию о финальном названии инструмента. В числе прочего предлагалось использовать префикс @redux.</p>
<p>Дискуссия продолжалась до середины октября 2018 года. Один из участников предложил вернуться к простому варианту <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">redux-starter-kit</code>. Проблема была в том, что кто-то уже опубликовал пакет с названием <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">redux-toolkit</code> на npm. Автор статьи на всякий случай обратился к владельцу пакета с ником @shotak и спросил, не может ли он подарить имя <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">redux-toolkit</code>. К удивлению автора, человек ответил в течение нескольких часов и согласился подарить имя. После этого автор опубликовал первую официальную версию инструмента.</p>
<h3 id="heading-3-10">Работа над функциональностью</h3>
<p>Функции <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createReducer</code> и <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createAction</code> уже были полезными. Но автор статьи хотел найти более простые инструменты.</p>
<p>Команда <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/17" rel="noopener noreferrer" target="_blank">обсуждала использование хелперов</a> из пакетов типа <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">redux-utilities</code>.</p>
<p>Автор публикации видел десятки подобных библиотек. Одна из библиотек, которая привлекла внимание автора — <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/17" rel="noopener noreferrer" target="_blank"><code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">autodux</code> Эрика Эллиотта</a>. Она позволяет пользователям автоматически генерировать создателей действий и типы действий на основе объекта с редьюсерами и префикса с именем.</p>
<p>Шон Ванг (Shawn Wang) <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/17#issuecomment-414091942" rel="noopener noreferrer" target="_blank">оценил</a> <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">autodux</code> и несколько других пакетов. Вот выдержка:</p>
<p>«Использование <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">autodux</code> даёт мощный набор создателей, селекторов и редьюсеров из коробки. Поэтому вы можете писать действия для бизнес-логики без дополнительных усилий. Мне нравится <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">autodux</code>».</p>
<p>В конце концов <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">autodux</code> стал источником вдохновения при создании самой полезной части Redux Toolkit: <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://redux-starter-kit.js.org/api/createslice" rel="noopener noreferrer" target="_blank">функции <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createSlice</code></a>.</p>
<p>Эрик Боуэр (Eric Bower) <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/pull/37" rel="noopener noreferrer" target="_blank">попробовал портировать</a> <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">autodaz</code> в <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createSlice</code>. Пулреквест застопорился. Автор статьи узнал о <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://www.reddit.com/r/javascript/comments/9fu82r/the_rise_of_immer_as_immutability_library_in_react/e5zmo3q/" rel="noopener noreferrer" target="_blank">библиотеке <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">robodux</code>, в которой используется Immer</a>. Он понял, что Эрик Боуэр решил сделать собственную версию библиотеки на TypeScript.</p>
<p>После обсуждения Боуэр согласился вернуть код в Redux Toolkit. Автор скомпилировал код библиотеки в чистый JavaScript и включил его в стартеркит.</p>
<h2 id="heading-2-11">Создание дорожной карты</h2>
<p>В начале декабря 2018 года с помощью Docusaurus <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/48" rel="noopener noreferrer" target="_blank">была опубликована документация Redux Toolkit</a>.</p>
<p>Автор статьи добавил в стартеркит дефолтные <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/48" rel="noopener noreferrer" target="_blank">мидлвары для определения мутаций и несериализируемых значений</a>, а также <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/pull/64" rel="noopener noreferrer" target="_blank">возможность трассировки стека</a>.</p>
<p>В конце января 2019 года автор статьи <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/82" rel="noopener noreferrer" target="_blank">открыл issue</a>, в котором предложил обсудить дорожную карту к выпуску redux Starter Kit 1.0, а также начал работать <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://blog.isquaredsoftware.com/2019/10/redux-starter-kit-1.0/" rel="noopener noreferrer" target="_blank">над способностью слайсов слушать другие действия</a>. Дэнис Вашингтон (Denis Washington) прокомментировал этот пулреквест. Он предложил <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/pull/83#pullrequestreview-194450952" rel="noopener noreferrer" target="_blank">обсудить целесообразность использования <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createSlice</code></a>.</p>
<p>В ответ автор публикации написал большой комментарий. Вот выдержки:</p>
<h4 id="heading-4-12">Цели</h4>
<p>Согласно README, в числе прочих источниками вдохновения для создателей Redux Toolkit являются Create-React-App и Apollo-Boost. Это пакеты, в которых есть дефолтные инструменты и абстракции над более сложными настраиваемыми инструментами. Цель существования этих пакетов — обеспечить быстрый старт при работе с библиотекой без необходимости погружаться в детали и принимать сложные решения. В то же время эти пакеты не привязывают разработчика к дефолтным инструментам и настройкам. Они очень полезны для новичков, которые не знают о всех доступных вариантах и о «правильных» выборах. Также пакеты полезны для опытных разработчиков, которым нужен простой инструмент, не требующий настройки окружения.</p>
<p>Я хочу, чтобы Redux Toolkit стал таким же инструментом для Redux.</p>
<h4 id="heading-4-13">Быстрый старт</h4>
<p>Я хочу, чтобы люди начинали работать с Redux как можно быстрее.</p>
<h4 id="heading-4-14">Упрощение основных способов применения и уход от бойлерплейтов</h4>
<p>Я хочу убрать острые углы:</p>
<ul>
<li>сократить код где это возможно;</li>
<li>предложить функции, которые выполняют очевидные действия, которые пользователи привыкли делать руками;</li>
<li>посмотреть, как люди используют Redux и как они хотят его использовать, и предложить «официальное» решение на основе предпочтений пользователей.</li>
</ul>
<p>Redux Toolkit должен вести пользователя к правильным настройкам, а также автоматически защищать от распространённых ошибок или предупреждать о них.</p>
<h4 id="heading-4-15">Отсутствие блокировки</h4>
<p>Когда кто-то начинает делать приложение на базе Redux Toolkit, я не хочу забирать у него возможность сделать что-то вручную в начале или в процессе работы.</p>
<p>С другой стороны, люди должны иметь возможность добавлять Redux Toolkit в существующие приложения. В таких случаях у разработчиков должна быть возможность внедрять стартеркит постепенно без необходимости переписывать код с нуля.</p>
<h4 id="heading-4-16">Стать очевидным дефолтным способом использования Redux</h4>
<p>В долгосрочной перспективе автор видит Redux Toolkit очевидным дефолтным способом использования Redux для большинства разработчиков. В качестве примера можно посмотреть на Create React App, который стал дефолтным очевидным способом использования React.</p>
<h4 id="heading-4-17">Отказ от попыток охватить все способы использования Redux, фокусировка на типичных подходах</h4>
<p>Есть много способов использования Redux. А в ядре Redux есть минимальный набор инструментов. Именно поэтому существуют тысячи библиотек и аддонов, который охватывают разные кейсы применения Redux. Стартеркит не может и не собирается охватывать все способы применения Redux.</p>
<h4 id="heading-4-18">Разумная поддержка TypeScript</h4>
<p>Автор хочет поддерживать специалистов, которые пишут на TypeScript, как и других специалистов, которые используют Redux специфическим способом. Поэтому в Redux Toolkit должна быть хорошая типизация. В то же время автор не хочет, чтобы «лучшая типизация» стала врагом «хорошей функциональности».</p>
<h2 id="heading-2-19">Работа с API</h2>
<p>Параллельно с этой дискуссией Дэнис Вашингтон конвертировал базу Redux Toolkit в TypeScript.</p>
<p>Автор публикации потратил несколько месяцев на работу с <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/react-redux/issues/1177" rel="noopener noreferrer" target="_blank">дорожной картой React-Redux 7</a>. Потом он работал с <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/react-redux/issues/1179" rel="noopener noreferrer" target="_blank">дизайном хуков API</a>.</p>
<p>Одновременно команда обсуждала разные вопросы, включая <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/91" rel="noopener noreferrer" target="_blank">расширение возможностей <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createSlice</code></a>. Рассматривалась функция <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">combineSlices</code>, а также автогенерируемый селектор функциональности.</p>
<p>К лету 2019 года автор завершил работу над React-Redux и начал писать <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/pull/164" rel="noopener noreferrer" target="_blank">документацию Redux Toolkit</a>. В начале сентября она была готова.</p>
<p>Тем временем Ленц Вебер (Lenz Weber) предложил несколько улучшений, включая <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/pull/149" rel="noopener noreferrer" target="_blank">внедрение колбэка <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">prepare</code> в <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createAction</code></a>, а также <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/pull/168" rel="noopener noreferrer" target="_blank">улучшение читабельности определений типов</a>.</p>
<p>С этого момента автор начал «допиливать» API Redux Toolkit. Появилась <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/pull/192" rel="noopener noreferrer" target="_blank">возможность кастомизации дефолтных мидлваро</a>в, а также <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/pull/209" rel="noopener noreferrer" target="_blank">включение редьюсеров в <code style="margin-bottom:var(--mantine-spacing-lg)" class="m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight m_e597c321 mantine-CodeHighlight-codeHighlight m_dfe9c588 mantine-InlineCodeHighlight-inlineCodeHighlight">createSlice</code></a>.</p>
<p>Было ещё много идей, но <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/82#issuecomment-542773046" rel="noopener noreferrer" target="_blank">автор решил</a>, что их можно реализовать после релиза. После этого официально вышел Redux Toolkit 1.0.</p>
<h2 id="heading-2-20">Будущее Redux Toolkit</h2>
<h3 id="heading-3-21">Обзор</h3>
<p>Оглядываясь на проделанную работу, можно сказать, что создателям удалось выполнить основные запланированные задачи. Redux Toolkit стал инструментом с такими характеристиками и возможностями:</p>
<ul>
<li>официальный пакет, в котором реализованы лучшие практики;</li>
<li>значительно упрощает работу с Redux;</li>
<li>сохраняет основные принципы Redux;</li>
<li>стартеркит можно добавить в проект на любой стадии реализации;</li>
<li>полезный инструмент как для новичков, так и для опытных разработчиков;</li>
<li>поддерживает TypeScript без увеличения сложности разработки.</li>
</ul>
<p>Автор активно рекламировал Redux Toolkit целый год. Ссылка на этот инструмент появилась на главной странице документации Redux. Также инструмент рекомендуется в разделе документации <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://redux.js.org/recipes/configuring-your-store" rel="noopener noreferrer" target="_blank">Configuring Your Store</a>.</p>
<p>Судя по <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://npm-stat.com/charts.html?package=redux-starter-kit&from=2018-07-01&to=2019-10-23" rel="noopener noreferrer" target="_blank">статистике загрузки пакета</a>, Redux Toolkit оказался востребованным инструментом. Его устанавливают более 100 тыс. раз в месяц, и график количества загрузок показывает рост. Автор получил много положительных откликов о Redux Toolkit.</p>
<style data-mantine-styles="inline">.__m__-_R_65derddmiub_{--carousel-slide-gap:var(--mantine-spacing-xs);--carousel-slide-size:80%;}@media(min-width: 36em){.__m__-_R_65derddmiub_{--carousel-slide-gap:var(--mantine-spacing-xl);--carousel-slide-size:50%;}}</style><div style="--carousel-control-size:calc(2.5rem * var(--mantine-scale));--carousel-controls-offset:var(--mantine-spacing-sm);margin-bottom:var(--mantine-spacing-lg);padding-block:var(--mantine-spacing-sm);background:var(--app-color-surface)" class="m_17884d0f mantine-Carousel-root responsiveClassName" data-orientation="horizontal" data-include-gap-in-size="true"><div class="m_39bc3463 mantine-Carousel-controls" data-orientation="horizontal"><button class="mantine-focus-auto m_64f58e10 mantine-Carousel-control m_87cf2631 mantine-UnstyledButton-root" type="button" data-inactive="true" data-type="previous" tabindex="-1"><svg viewBox="0 0 15 15" fill="none" xmlns="http://www.w3.org/2000/svg" style="transform:rotate(90deg);width:calc(1rem * var(--mantine-scale));height:calc(1rem * var(--mantine-scale));display:block"><path d="M3.13523 6.15803C3.3241 5.95657 3.64052 5.94637 3.84197 6.13523L7.5 9.56464L11.158 6.13523C11.3595 5.94637 11.6759 5.95657 11.8648 6.15803C12.0536 6.35949 12.0434 6.67591 11.842 6.86477L7.84197 10.6148C7.64964 10.7951 7.35036 10.7951 7.15803 10.6148L3.15803 6.86477C2.95657 6.67591 2.94637 6.35949 3.13523 6.15803Z" fill="currentColor" fill-rule="evenodd" clip-rule="evenodd"></path></svg></button><button class="mantine-focus-auto m_64f58e10 mantine-Carousel-control m_87cf2631 mantine-UnstyledButton-root" type="button" data-inactive="true" data-type="next" tabindex="-1"><svg viewBox="0 0 15 15" fill="none" xmlns="http://www.w3.org/2000/svg" style="transform:rotate(-90deg);width:calc(1rem * var(--mantine-scale));height:calc(1rem * var(--mantine-scale));display:block"><path d="M3.13523 6.15803C3.3241 5.95657 3.64052 5.94637 3.84197 6.13523L7.5 9.56464L11.158 6.13523C11.3595 5.94637 11.6759 5.95657 11.8648 6.15803C12.0536 6.35949 12.0434 6.67591 11.842 6.86477L7.84197 10.6148C7.64964 10.7951 7.35036 10.7951 7.15803 10.6148L3.15803 6.86477C2.95657 6.67591 2.94637 6.35949 3.13523 6.15803Z" fill="currentColor" fill-rule="evenodd" clip-rule="evenodd"></path></svg></button></div><div class="m_a2dae653 mantine-Carousel-viewport" data-type="media"><div class="m_fcd81474 mantine-Carousel-container __m__-_R_65derddmiub_" data-orientation="horizontal"><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/stoit-li-uchit-javascript-perspektivy-situatsiya-na-rynke-truda-mneniya-ekspertov"><div style="padding-top:0rem;height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="margin-bottom:var(--mantine-spacing-sm)" class="m_599a2148 mantine-Card-section" data-first-section="true"><div style="--ar-ratio:2" class="m_71ac47fc mantine-AspectRatio-root"><img class="m_9e117634 mantine-Image-root" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MTcwMywicHVyIjoiYmxvYl9pZCJ9fQ==--aa69869c906a992728ba7c26886fa1b715024424/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/%D1%8F%D0%B7%D1%8B%D0%BA%D0%BF%D1%80%D0%BE%D0%B3%D1%80%D0%B0%D0%BC%D0%BC%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D1%8Fjavascript-01.png" loading="lazy" alt="Язык программирования JavaScript: где его используют и почему он популярен"/></div></div><p style="margin-bottom:var(--mantine-spacing-xs);font-size:var(--mantine-font-size-lg);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Язык программирования JavaScript: где его используют и почему он популярен</p><p style="margin-bottom:auto" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Рассказываем, что это за язык программирования — JavaScript, где его используют, насколько он поп...</p><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-lg);font-size:var(--mantine-font-size-sm)" class="m_4081bf90 mantine-Group-root">4 октября 2023 г.<p style="font-size:inherit" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/zachem-izuchat-node-js-ili-o-perspektivah-bekenda-na-javascript"><div style="padding-top:0rem;height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="margin-bottom:var(--mantine-spacing-sm)" class="m_599a2148 mantine-Card-section" data-first-section="true"><div style="--ar-ratio:2" class="m_71ac47fc mantine-AspectRatio-root"><img class="m_9e117634 mantine-Image-root" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MTY4MywicHVyIjoiYmxvYl9pZCJ9fQ==--8e5dd13a34bb6d61f1c0db87a25e6e64d0e237fb/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/node%20_new-01.png" loading="lazy" alt="Все про Node.js: зачем писать бэкенд на JavaScript и сколько за это платят"/></div></div><p style="margin-bottom:var(--mantine-spacing-xs);font-size:var(--mantine-font-size-lg);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Все про Node.js: зачем писать бэкенд на JavaScript и сколько за это платят</p><p style="margin-bottom:auto" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Рассказываем, как устроен бэкенд на JavaScript, стоит ли изучать Node.js, когда есть PHP и Python...</p><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-lg);font-size:var(--mantine-font-size-sm)" class="m_4081bf90 mantine-Group-root">28 июля 2023 г.<p style="font-size:inherit" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></a></div></div><div class="m_d98df724 mantine-Carousel-slide" data-orientation="horizontal"><div tabindex="0" style="cursor:pointer;height:100%"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/zachem-izuchat-php-reyting-perspektivy-sfery-primeneniya"><div style="padding-top:0rem;height:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root" data-with-border="true"><div style="margin-bottom:var(--mantine-spacing-sm)" class="m_599a2148 mantine-Card-section" data-first-section="true"><div style="--ar-ratio:2" class="m_71ac47fc mantine-AspectRatio-root"><img class="m_9e117634 mantine-Image-root" src="https://hexlet.io/rails/active_storage/representations/proxy/eyJfcmFpbHMiOnsiZGF0YSI6MTcxNSwicHVyIjoiYmxvYl9pZCJ9fQ==--483469e30822f45e5959af1ca8e34c28c3fe28db/eyJfcmFpbHMiOnsiZGF0YSI6eyJmb3JtYXQiOiJ3ZWJwIiwicmVzaXplX2FuZF9wYWQiOls3MDgsMzU0XSwic2F2ZXIiOnsicXVhbGl0eSI6ODV9fSwicHVyIjoidmFyaWF0aW9uIn19--324dc52aa55ebe818c2a887ebcb832b9ad1c0381/%D1%81%D1%87%D0%B5%D0%B3%D0%BE%D0%BD%D0%B0%D1%87%D0%B0%D1%82%D1%8C%D0%B8%D0%B7%D1%83%D1%87%D0%B0%D1%82%D1%8Cphp-2-01.png" loading="lazy" alt="Язык программирования PHP: рейтинг, сферы применения, прогнозы экспертов"/></div></div><p style="margin-bottom:var(--mantine-spacing-xs);font-size:var(--mantine-font-size-lg);font-weight:bold" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Язык программирования PHP: рейтинг, сферы применения, прогнозы экспертов</p><p style="margin-bottom:auto" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Рассказываем, зачем и где учить PHP, где его применяют, сколько зарабатывают PHP-разработчики, а ...</p><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap;margin-top:var(--mantine-spacing-lg);font-size:var(--mantine-font-size-sm)" class="m_4081bf90 mantine-Group-root">24 марта 2023 г.<p style="font-size:inherit" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></div></a></div></div></div></div></div>
<h2 id="heading-2-22">Следующие шаги</h2>
<p>В настоящее время <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux/issues/3313#issuecomment-450601554" rel="noopener noreferrer" target="_blank">планируется</a> масштабная работа по модернизации документации Redux. В рамках модернизации будет созданы обучающие материалы по Redux Toolkit. Также информация о стартерките будет добавлена в гайды и туториалы. Пользователи узнают о простых способах работы с Redux, например, об избегании использования констант действия или об <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/erikras/ducks-modular-redux" rel="noopener noreferrer" target="_blank">использовании утиного паттерна при организации логики</a>. Возможно, в Redux Toolkit добавят <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://github.com/reduxjs/redux-starter-kit/issues/76" rel="noopener noreferrer" target="_blank">возможность определять асинхронные побочные эффекты в слайсах</a>.</p>
<p><em>Адаптированный перевод статьи <a style="text-decoration:underline" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="https://blog.isquaredsoftware.com/2019/10/redux-starter-kit-1.0/" rel="noopener noreferrer" target="_blank">Idiomatic Redux: Redux Toolkit 1.0</a> by Mark Erikson. Мнение администрации «Хекслета» может не совпадать с мнением автора оригинальной публикации.</em></p></div><div class=""><div style="--group-gap:var(--mantine-spacing-md);--group-align:center;--group-justify:space-between;--group-wrap:wrap;margin-bottom:var(--mantine-spacing-lg)" class="m_4081bf90 mantine-Group-root"><div class="m_4451eb3a mantine-Center-root" data-inline="true"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;margin-inline-end:var(--mantine-spacing-xs);color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-user "><path d="M8 7a4 4 0 1 0 8 0a4 4 0 0 0 -8 0"></path><path d="M6 21v-2a4 4 0 0 1 4 -4h4a4 4 0 0 1 4 4v2"></path></svg></div><p style="margin-inline-end:var(--mantine-spacing-xl)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Дмитрий Дементий</p><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">6 лет назад</p></div><div style="align-items:center" class="m_8bffd616 mantine-Flex-root __m__-_R_5dirddmiub_"><a style="display:inline-flex" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/idiomaticheskiy-redux-redux-starter-kit-1-0/votes"><div style="--ti-size:var(--ti-size-sm);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;margin-inline-end:var(--mantine-spacing-xs);color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="sm"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-thumb-up "><path d="M7 11v8a1 1 0 0 1 -1 1h-2a1 1 0 0 1 -1 -1v-7a1 1 0 0 1 1 -1h3a4 4 0 0 0 4 -4v-1a2 2 0 0 1 4 0v5h3a2 2 0 0 1 2 2l-1 5a2 3 0 0 1 -2 2h-7a3 3 0 0 1 -3 -3"></path></svg></div></a><p class="mantine-focus-auto m_b6d8b162 mantine-Text-root">8</p></div></div></div><div style="background-color:var(--mantine-color-indigo-light);border:calc(0.0625rem * var(--mantine-scale)) solid transparent;padding:var(--mantine-spacing-xl)" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root"><p style="margin-bottom:var(--mantine-spacing-sm);font-size:var(--mantine-font-size-h4)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Читайте также:</p><ul style="margin-inline-start:var(--mantine-spacing-lg)" class="m_abbac491 mantine-List-root"><li style="margin-bottom:var(--mantine-spacing-sm)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><a style="color:inherit" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/stoit-li-uchit-javascript-perspektivy-situatsiya-na-rynke-truda-mneniya-ekspertov">Язык программирования JavaScript: где его используют и почему он популярен</a></span></div></li><li style="margin-bottom:var(--mantine-spacing-sm)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><a style="color:inherit" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/zachem-izuchat-node-js-ili-o-perspektivah-bekenda-na-javascript">Все про Node.js: зачем писать бэкенд на JavaScript и сколько за это платят</a></span></div></li><li style="margin-bottom:var(--mantine-spacing-sm)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><a style="color:inherit" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/blog/posts/zachem-izuchat-php-reyting-perspektivy-sfery-primeneniya">Язык программирования PHP: рейтинг, сферы применения, прогнозы экспертов</a></span></div></li></ul></div><div style="margin-block:var(--mantine-spacing-xl)" class="m_3eebeb36 mantine-Divider-root" data-orientation="horizontal" role="separator"></div></div><div></div></div><style data-mantine-styles="inline">.__m__-_R_lmiub_{--col-flex-grow:auto;--col-flex-basis:100%;--col-max-width:100%;}@media(min-width: 48em){.__m__-_R_lmiub_{--col-flex-grow:auto;--col-flex-basis:16.666666666666668%;--col-max-width:16.666666666666668%;}}@media(min-width: 62em){.__m__-_R_lmiub_{--col-flex-grow:auto;--col-flex-basis:33.333333333333336%;--col-max-width:33.333333333333336%;}}</style><div class="m_96bdd299 mantine-Grid-col __m__-_R_lmiub_ mantine-visible-from-md"><div style="background-color:var(--mantine-color-indigo-light);border:calc(0.0625rem * var(--mantine-scale)) solid transparent;margin-bottom:var(--mantine-spacing-xl);padding:var(--mantine-spacing-xl);width:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root"><div style="margin-bottom:var(--mantine-spacing-md)" class="m_4451eb3a mantine-Center-root" data-inline="true"><p style="font-size:var(--mantine-font-size-h4)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Категории</p></div><ul class="m_abbac491 mantine-List-root"><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Мотивация">Мотивация</button></span></div></li><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Истории успеха">Истории успеха</button></span></div></li><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Дневник студента">Дневник студента</button></span></div></li><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Код">Код</button></span></div></li><li style="margin-bottom:var(--mantine-spacing-xs)" class="m_abb6bec2 mantine-List-item" data-with-icon="true"><div class="m_75cd9f71 mantine-List-itemWrapper"><span class="m_60f83e5b mantine-List-itemIcon"><div class="m_4451eb3a mantine-Center-root"><div style="--ti-size:var(--ti-size-xs);--ti-bg:transparent;--ti-color:var(--mantine-color-indigo-light-color);--ti-bd:calc(0.0625rem * var(--mantine-scale)) solid transparent;color:inherit" class="m_7341320d mantine-ThemeIcon-root" data-variant="transparent" data-size="xs"><svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round" class="tabler-icon tabler-icon-chevron-compact-right "><path d="M11 4l3 8l-3 8"></path></svg></div></div></span><span class="mantine-List-itemLabel"><button style="color:inherit;text-decoration:underline" class="mantine-focus-auto m_87cf2631 mantine-UnstyledButton-root" type="button" aria-label="Карьера">Карьера</button></span></div></li></ul></div><div style="justify-content:end;margin-top:0rem;position:sticky;top:calc(5rem * var(--mantine-scale))" class="m_8bffd616 mantine-Flex-root __m__-_R_5dlmiub_"><div tabindex="0" style="cursor:pointer"><a style="text-decoration:none" class="mantine-focus-auto m_849cf0da m_b6d8b162 mantine-Text-root mantine-Anchor-root" data-underline="hover" href="/courses_web_development?promo_name=program_category&promo_position=blog_post&promo_creative=card&promo_type=card"><div style="background-color:var(--mantine-color-default);border:calc(0.0625rem * var(--mantine-scale)) solid var(--mantine-color-default-border);padding-inline:var(--mantine-spacing-xl);padding-top:var(--mantine-spacing-xl);padding-bottom:var(--mantine-spacing-xs);width:100%" class="m_e615b15f mantine-Card-root m_1b7284a3 mantine-Paper-root"><div class="m_4451eb3a mantine-Center-root" data-inline="true"><p style="font-size:var(--mantine-font-size-h4)" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Курсы по веб-разработке</p></div><img class="m_9e117634 mantine-Image-root" src="/vite/assets/development-BVihs_d5.png"/><p style="margin-bottom:var(--mantine-spacing-xs);text-align:right" class="mantine-focus-auto m_b6d8b162 mantine-Text-root">Посмотреть →</p></div></a></div></div></div></div></div></div></div>
</main>
<footer class="bg-dark fw-light text-light px-3 py-5">
<div class="row small">
<div class="col-12 col-sm-6 col-md-3">
<div class="h5 mb-3">Хекслет</div>
<ul class="list-unstyled">
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/about">О нас</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/testimonials">Отзывы</a>
</li>
<li>
<span class="nav-link link-light py-1 ps-0 external-link" data-href="https://b2b.hexlet.io" role="button">Корпоративное обучение</span>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/blog">Блог</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/qna">Вопросы и ответы</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/glossary">Глоссарий</a>
</li>
<li>
<span class="nav-link link-light py-1 ps-0 external-link" data-href="https://help.hexlet.io" data-target="_blank" role="button">Справка</span>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" target="_blank" rel="noopener noreferrer" href="/map">Карта сайта</a>
</li>
</ul>
</div>
<div class="col-12 col-sm-6 col-md-3">
<div class="h5 fw-normal mb-3">Направления</div>
<ul class="list-unstyled">
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_devops">DevOps
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_data_analytics">Аналитика
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_backend_development">Бэкенд
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_programming">Программирование
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_testing">Тестирование
</a></li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/courses_front_end_dev">Фронтенд
</a></li>
</ul>
</div>
<div class="col-12 col-sm-6 col-md-3">
<div class="h5">Профессии</div>
<ul class="list-unstyled">
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/devops-engineer-from-scratch">DevOps-инженер с нуля</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/go">Go-разработчик</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/java">Java-разработчик</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/python">Python-разработчик </a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/data-analytics">Аналитик данных</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/qa-engineer">Инженер по ручному тестированию</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/php">РНР-разработчик</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/frontend">Фронтенд-разработчик</a>
</li>
</ul>
</div>
<div class="col-12 col-sm-6 col-md-3">
<div class="h5">Навыки</div>
<ul class="list-unstyled">
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/python-django-developer">Django</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/docker">Docker</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/php-laravel-developer">Laravel</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/postman">Postman</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/js-react-developer">React</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/js-rest-api">REST API в Node.js</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/spring-boot">Spring Boot</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/programs/typescript">Typescript</a>
</li>
</ul>
</div>
</div>
<hr>
<div class="row">
<div class="col-12 col-sm-4 col-md-2">
<div class="fs-4">
<ul class="list-unstyled d-flex">
<li class="me-3">
<a aria-label="Telegram" target="_blank" class="link-light" rel="noopener noreferrer nofollow" href="https://t.me/hexlet_ru"><span class="bi bi-telegram"></span>
</a></li>
<li>
<a aria-label="Youtube" target="_blank" class="link-light" rel="noopener noreferrer nofollow" href="https://www.youtube.com/user/HexletUniversity"><span class="bi bi-youtube"></span>
</a></li>
</ul>
</div>
<div class="mb-2 d-flex flex-column">
<a class="link-light text-decoration-none" rel="nofollow" href="mailto:support@hexlet.io">support@hexlet.io</a>
<a class="link-light text-decoration-none py-2" target="_blank" href="https://t.me/hexlet_help_bot">t.me/hexlet_help_bot</a>
</div>
<ul class="list-unstyled d-flex">
<li class="me-3">
<span class="link-light text-decoration-none opacity-50 x-font-size-18 external-link" rel="nofollow" data-href="https://hexlet.io/locale/switch?new_locale=en" data-target="_self" role="button"><span class="my-auto">EN</span>
</span></li>
<li class="me-3">
<span class="link-light text-decoration-none opacity-50 x-font-size-18 opacity-100 external-link" rel="nofollow" data-href="https://ru.hexlet.io/locale/switch?new_locale=ru" data-target="_self" role="button"><span class="my-auto">RU</span>
</span></li>
<li class="me-3">
<span class="link-light text-decoration-none opacity-50 x-font-size-18 external-link" rel="nofollow" data-href="https://kz.hexlet.io/locale/switch?new_locale=kz" data-target="_self" role="button"><span class="my-auto">KZ</span>
</span></li>
</ul>
</div>
<div class="col-12 col-sm-4 col-md-3">
<ul class="list-unstyled fs-4">
<li class="mb-3">
<a class="link-light text-decoration-none" href="tel:8%20800%20100%2022%2047">8 800 100 22 47</a>
<span class="d-block opacity-50 small">бесплатно по РФ</span>
</li>
<li>
<a class="link-light text-decoration-none" href="tel:%2B7%20495%20085%2021%2062">+7 495 085 21 62</a>
<span class="d-block opacity-50 small">бесплатно по Москве</span>
</li>
</ul>
</div>
<div class="col-12 col-sm-4 col-md-3">
<div class="small mb-3">Образовательные услуги оказываются на основании Л035-01298-77/01989008 от 14.03.2025</div>
<ul class="list-unstyled small">
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/legal">Правовая информация</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/offer">Оферта</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/license">Лицензия</a>
</li>
<li>
<a class="nav-link link-light py-1 ps-0" href="/pages/contacts">Контакты</a>
</li>
</ul>
</div>
<div class="col-12 col-sm-12 col-md-4 small">
<div class="mb-2">
<div>ООО «<a href="/" class="text-decoration-none link-light">Хекслет Рус</a>»</div>
<div>108813 г. Москва, вн.тер.г. поселение Московский,</div>
<div>г. Московский, ул. Солнечная, д. 3А, стр. 1, помещ. 20Б/3</div>
<div>ОГРН 1217300010476</div>
<div>ИНН 7325174845</div>
</div>
<hr>
<div>АНО ДПО «<a href="/" class="text-decoration-none link-light">Учебный центр «Хекслет</a>»</div>
<div>119331 г. Москва, вн. тер. г. муниципальный округ</div>
<div>Ломоносовский, пр-кт Вернадского, д. 29</div>
<div>ОГРН 1247700712390</div>
<div>ИНН 7736364948</div>
</div>
</div>
</footer>
<div id="root-assistant-offcanvas"></div>
<script src="/vite/assets/assistant-Bukl1lYy.js" crossorigin="anonymous" type="module"></script><link rel="modulepreload" href="/vite/assets/chunk-DsPFFUou.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/init-BrRXra1y.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/ErrorFallbackBlock-naDSYSy9.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/MarkdownBlock-DbyKWoR_.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/gon-D3e4yh1x.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/mantine-CGMYrt2Y.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/shiki-V011pkdv.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/utils-DRqSHbQE.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/routes-CCH8ilKF.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/lib-XR8Qr8kR.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/dist-GCHh59xr.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/Box-B5-OOzBf.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/notifications.store-C-3AFSMn.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/useIsomorphicEffect-HJ6VK0D3.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/lib-KSp6QbZ0.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/axios-BEvgo0ym.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/classnames-l6ipYlLR.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/dayjs.min-BkKovM-s.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/debounce-jMQ_Cf4f.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/i18next-BlSq9s7B.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/client-U9M77rxp.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/react-dom-DaLxUz_h.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/useTranslation-Bx1Cdrkz.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/compiler-runtime-6XxiPFnt.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/jsx-runtime-CwjcCKJi.js" as="script" crossorigin="anonymous">
<link rel="modulepreload" href="/vite/assets/react-CkL4ZRHB.js" as="script" crossorigin="anonymous">
<script defer src="https://static.cloudflareinsights.com/beacon.min.js/v67327c56f0bb4ef8b305cae61679db8f1769101564043" integrity="sha512-rdcWY47ByXd76cbCFzznIcEaCN71jqkWBBqlwhF1SY7KubdLKZiEGeP7AyieKZlGP9hbY/MhGrwXzJC/HulNyg==" data-cf-beacon='{"version":"2024.11.0","token":"d11015b65d11429ea6b4a2ef37dd7e0b","server_timing":{"name":{"cfCacheStatus":true,"cfEdge":true,"cfExtPri":true,"cfL4":true,"cfOrigin":true,"cfSpeedBrain":true},"location_startswith":null}}' crossorigin="anonymous"></script>
</body>
</html>