Good day, Khabrovsk!
I want to talk about how I recently learned about certain “hooks” in React. They appeared relatively recently, in the version [16.8.0] of February 6, 2019 (which, according to the development speeds of FrontEnd, is already a very long time ago)
After reading the documentation, I focused on the useReducer hook and immediately asked myself the question: "This thing can completely replace Redux !?" I spent several evenings on experiments and now I want to share the results and my conclusions.
Do I need to replace Redux with useContext + useReducer?
For the impatient - immediately conclusions
Per:
- You can use hooks (useContext + useReducer) instead of Redux in small applications (where there is no need for large combined Reducers). In this case, Redux may indeed be redundant.
Against:
- A large amount of code has already been written on a bunch of React + Redux and to rewrite it to hooks (useContext + useReducer) seems to me inappropriate, at least for now.
- Redux is a proven library, hooks are an innovation, their interfaces and behavior may change in the future.
- In order to make using useContext + useReducer really convenient, you will have to write some bikes.
The conclusions are the personal opinion of the author and do not claim to be unconditional truth - if you do not agree, I will be glad to see your constructive criticism in the comments.
Let's try to figure it out
Let's start with a simple example.
(reducer.js)
import React from "react"; export const ContextApp = React.createContext(); export const initialState = { app: { test: 'test_context' } }; export const testReducer = (state, action) => { switch(action.type) { case 'test_update': return { ...state, ...action.payload }; default: return state } };
So far, our reducer looks exactly the same as in Redux
(app.js)
import React, {useReducer} from 'react' import {ContextApp, initialState, testReducer} from "./reducer.js"; import {IndexComponent} from "./IndexComponent.js" export const App = () => {
(IndexComponent.js)
import React, {useContext} from "react"; import {ContextApp} from "./reducer.js"; export function IndexComponent() {
This is the simplest example in which we simply update write new data to a flat (without nesting) reducer
In theory, you can even try to write like this:
(reducer.js)
... export const testReducer = (state, data) => { return { ...state, ...data } ...
(IndexComponent.js)
... return (
If we do not have a large and simple application (which is rarely the case in reality), then you can not use type and always manage the reducer update directly from the action. By the way, at the expense of updates, in this case we only wrote new data in reducer, but what if we have to change one value in a tree with several levels of nesting?
More complicated now
Let's look at the following example:
(IndexComponent.js)
... return (
(reducer.js)
... export const initialState = { tree_1: { tree_2_1: { tree_3_1: 'tree_3_1', tree_3_2: 'tree_3_2' }, tree_2_2: { tree_3_3: 'tree_3_3', tree_3_4: 'tree_3_4' } } }; export const testReducer = (state, callback) => {
Okay, we figured out the tree update too. Although in this case it is already better to return to using types inside testReducer and update the tree according to a certain type of action. Everything is like in Redux, only the resulting bundle is slightly smaller [8].
Asynchronous operations and dispatch
But is everything all right? What happens if we go in to use asynchronous operations?
To do this, we will have to define our own dispatch. Let's try!
(action.js)
export const actions = { sendToServer: function ({dataForServer}) {
(IndexComponent.js)
const [state, _dispatch] = useReducer(AppReducer, AppInitialState);
It seems that everything is okay too, but now we have a lot of nested callbacks , which is not very cool, if we just want to change the state without creating an action function, we will have to write a construction of this kind:
(IndexComponent.js)
... dispatch( (dispatch) => dispatch(state => { return { {dataForServer: 'data'} } }) ) ...
It turns out something scary, right? For a simple update of the data, I would very much like to write something like this:
(IndexComponent.js)
... dispatch({dataForServer: 'data'}) ...
To do this, you will have to change the Proxy for the dispatch function that we created earlier
(IndexComponent.js)
const [state, _dispatch] = useReducer(AppReducer, AppInitialState);
Now we can pass both an action function and a simple object to dispatch.
But! With a simple transfer of the object, you must be careful, you may be tempted to do this:
(IndexComponent.js)
... dispatch({ tree: {
Why is this example bad? By the fact that by the time this dispatch was processed, state could have been updated through another dispatch, but these changes have not yet reached our component and in fact we are using an old state instance that will overwrite everything with old data.
For this reason, such a method becomes hardly suitable anywhere, only for updating flat reducers in which there is no nesting and you do not need to use state to update nested objects. In reality, reducers are rarely perfectly flat, so I would advise you not to use this method at all and only update data through actions.
(action.js)
...
Conclusions:
- It was an interesting experience, I strengthened my academic knowledge and learned new features of the reaction
- I will not use this approach in production (at least in the next six months). For the reasons already described above (this is a new feature, and Redux is a proven and reliable tool) + I have no performance problems to chase after the milliseconds that you can win by abandoning the editor [8]
I will be glad to know, in the comments, the opinion of colleagues from the front-end part of our Habrosobschestva!
References: