Why you shouldn’t use Redux?

What is  Redux?

Redux is an open-source JavaScript library for managing application state using a simple constrained  API designed as a predictable container for application state. It works similar to reducing function, a programming concept.

People have been using Redux for years . Redux was a revolutionary technology in React ecosystem. It allowed us to create a global “vault” with immutable data and fixed the prop drilling problem in our component tree. It is still a great tool for sharing immutable data in the application , which additionally scales well. Redux helps you write applications that behave consistently, run in different environments (client, server  and native) and are easy to test.

Why should I not use Redux then?

With all these great advantages, you might be thinking why shouldn’t we use Redux?
It is often a triumph of form over content , and contains  a large amount of boilerplate.
The main problem we face when using  Redux and similar state management libraries is treating it as a cache for our backend state.

We fetch the data , add it to our store with reducer and action, and then re-fetch from time to time, hoping that we are up to date. We rely too much on Redux and we use it as a universal solution to all our problems.
When you’re developing a smaller application, it is not a bad idea to abandon Redux library altogether.

Often a large amount of code is required to make state management transparent, consistent and extensible. Additionally, the entry threshold in Redux is quite high, and understanding how action reductors and  dispatches work can be quite overwhelming for beginners.

Great features come with a lot of complexity that can be an unnecessary burden on your project, especially if it is small.

When is it worth using  Redux?

There are still certain circumstances, when application may actually have a huge amount of client-only synchronous states (such as visualization application or a music production application), in which case you will likely need a client state manager.

You should use  React Redux especially if the UI is updated frequently, when many components have to respond to  one action and the actions themselves are very complicated – then Redux is a good idea. Many companies still consider the Redux library as a mandatory technology that every developer should understand –it is visible, especially  in the requirements of the labor market.

What to use instead of  Redux?

React Query

The first very comprehensive and flexible state management library that comes to mind is React Query.

To make it easy to see the difference between  React Query and  Redux I will introduce the principles of both in the code. I implemented a simple  TODO retrieved from the server using both methods, using pure JavaScript,  React Hooks and  Axios. It is also worth noting that  Redux and  React Query can be used simultaneously, one library does not exclude the other.

First, the  Redux implementation:

import React, { useEffect } from "react";
import { useSelector, useDispatch } from "react-redux";
import axios from 'axios';
const SET_TODOS = "SET_TODOS";
export const rootReducer = (state = { todos: [] }, action) => {
switch (action.type) {
case SET_TODOS:
return { ...state, todos: action.payload };
default:
return state;
}
};
export const App = () => {
const todos = useSelector((state) => state.todos);
const dispatch = useDispatch();
useEffect(() => {
const fetchPosts = async () => {
const { data } = await axios.get("/api/todos");
dispatch({
type: SET_TODOS,
payload: data}
);
};
fetchPosts();
}, []);
return (
<ul>{todos.length > 0 && todos.map((todo) => <li>{todo.text}</li>)}</ul>
);
};

Note that the above code doesn’t even support re-fetching, caching or invalidation . It just loads the data and stores it in the global store when the component is loaded. If you want to cache your data, there is still a lot of work to do. Needless to say, there will be more boilerplate to come.

Here’s the same example , already using React Query:

import React from "react";
import { useQuery } from "react-query";
import axios from "axios";
const fetchTodos = () => {
const { data } = axios.get("/api/todos");
return data;
};
const App = () => {
const { data } = useQuery("todos", fetchTodos);
return data ? (
<ul>{data.length > 0 && data.map((todo) => <li>{todo.text}</li>)}</ul>
) : null;
};

Much better.

By default, this example includes re-fetching, caching and invalidation with quite reasonable defaults. You can also set the caching configuration on a global level and forget about it completely – in most cases it will behave the way you want. For more information on how it works under the hood, see the React Query documentation. There are tons of configuration option available, which only illustrates the potential of this library.

Wherever you need this data, you can now use the useQuery hook with a unique key set ( „todos” in this case) and the asynchronous call used to retrieve the data. If the function is asynchronous , the implementation doesn’t matter — you can just easily use the Fetch API instead of  Axios.

To change the state of our backend, React Query provides the useMutation hook.

For most applications, the truly  globally available  client state that remains after the asynchronous code has been migrated from Redux to React Query is usually very small.

Unstated

Here’s an example of state management using the Unstated library:

import { useState } from "react";
import { createContainer } from "unstated-next";
function useCounter() {
let [count, setCount] = useState(0);
let decrement = () => setCount(count - 1);
let increment = () => setCount(count + 1);
return { count, decrement, increment };
}
let Counter = createContainer(useCounter);
function CounterDisplay() {
let counter = Counter.useContainer();
return (
<div>
<button onClick={counter.decrement}>-</button>
<p>You clicked {counter.count} times</p>
<button onClick={counter.increment}>+</button>
</div>
);
}
function App() {
return (
<Counter.Provider>
<CounterDisplay />
<CounterDisplay />
</Counter.Provider>
);
}

From a code transparency perspective, it’s a great library. It uses  React hooks for all its state management logic.

If you are using TypeScript (which I strongly encourage you to do, if you haven’t tried it yet), it also has the advantage that the built-in TypeScript inference works better. As long as your non-standard hook is typed, everything else will just work.

But what are the real advantages of using  Unstated that trump  Redux?

  • It is smaller It is approx.. 40 times smaller.
  • It is faster. You can “component” the performance problem.
  • It is easier to learn. You will need to know  React & Context, Hooks from the start, use them they are really great.
  • It is simpler in terms of integration. Integrate one component at a time and easily integrate with any React library.
  • It is easier to test. Testing reducers is a waste of  your time, make testing your components easier.
  • It is easier to check types. Designed to bring out all the advantages of typing.
  • It is minimalistic. It’s just  React.

Redux Toolkit

Although it is not a separate technology, but only an add-on library to Redux, Redux Toolkit is a set of tools for efficient  Redux creation. This is supposed to be a standard way to write  Redux logic , and the developers of  Redux strongly recommend using it.

It includes several utility functions that simplify the most common Redux use cases, including store configuration, defining reducers, immutable update logic, and even creating entire „slices” of the state at once, without manually typing any creator or action types. It also includes the most commonly used  Redux add-ons such as Redux Thunk for async logic and  Reselect for writing selector functions, so you can use them right away.

Redux Toolkit offers a configureStore feature that provides useful defaults such as standard extensions when creating a store. The function is waiting for the configuration object. The configuration object tries to map the parameters of the createStore  function in a more understandable way.

The example of the configuration of a  Redux store with various reducers, middleware, support for development tools ,the initial state and the enchancer can be seen in the example below:

const reducer = {
todos: todosReducer,
visibility: visibilityReducer
}
const middleware = [...getDefaultMiddleware(), logger]
const preloadedState = {
todos: [
{
text: 'Eat food',
completed: true
}
],
visibilityFilter: 'SHOW_COMPLETED'
}
const store = configureStore({
reducer,
middleware,
devTools: process.env.NODE_ENV !== 'production',
preloadedState,
enhancers: [reduxBatch]
})

In order to create an action in Redux, you usually need a constant for the type and an ActionCreator that uses that constant. This leads to more boilerplate. The Redux Toolkit createAction method combines these two steps into one. Here is an example of a normal ActionCreator and how you can create it with the Redux toolset.

Without Toolkit:

const INCREMENT = 'counter/increment'
function increment(amount) {
return {
type: INCREMENT,
payload: amount
}
}
const action = increment(3)
// { type: 'counter/increment', payload: 3 }

With Toolkit:

 

const increment = createAction('counter/increment')
action = increment(3)
// returns { type: 'counter/increment', payload: 3 }

This feature helps to remove most of the standard code. One disadvantage is that you can no longer determine what format payload should be in. With the generated actions payload is not defined in more detail and you need to check the reductor that needs to be passed as payload.

If you want to go a step further, actions for the store could be generated automatically. For this purpose Redux Toolkit offers the createSlice method. The method requires slice name, initial state and a set of reducer functions as parameters.

The createSlice method then returns a set of generated elements – ActionCreators that match the keys in the Reducer module. The generated actions can receive the appropriate payload and can be easily used. However, generating these actions reaches its limit relatively quickly. It is no longer possible to map asynchronous actions, for example with Redux Thunk.
The following example shows how to use functions and the resulting action creators:

const user = createSlice({
name: 'user',
initialState: { name: '', age: 20 },
reducers: {
setUserName: (state, action) => {
state.name = action.payload
},
increment: (state, action) => {
state.age += 1
}
},
})
const reducer = combineReducers({
user: user.reducer
})
const store = createStore(reducer)
store.dispatch(user.actions.increment())
// -> { user: {name : '', age: 21} }
store.dispatch(user.actions.increment())
// -> { user: {name: '', age: 22} }
store.dispatch(user.actions.setUserName('eric'))
// -> { user: { name: 'eric', age: 22} }

The  createSlice method can significantly reduce the required boilerplate  for some stores, if only because you do not need to return the entire state, you only need to overwrite a single property of a given state. However, it is not flexible and it should be checked whether it is suitable for a specific application.

Redux Toolkit includes a number of features that can simplify working with Redux. It covers many standard cases by default, but you still can configure it for more specific tasks. The  Redux Toolkit package contains a collection of libraries that are already widely used and work well together. Thanks to these libraries and new functions of the Redux Toolkit many cases of  boilerplate can be avoided and the syntax becomes a little clearer and easier to understand.

What about  caching?

The library that is very good at  caching is React Query. The components are kept in the cache, so when we receive the data from the server, they are displayed almost instantaneously, even if we refresh the DOM. This is crucial for any project where multiple components displayed simultaneously are downloaded from the server. Caching prevents overloading of each element, thus significantly improving performance and overall User Experience.

I have just created my new project – I’m installing Redux!

Many people take Redux  for granted when they start working on their new project. Don’t forget that you can apply ideas from Redux without actually using it. For example consider the  React component with local state:

import React, { useState } from "react";
function Counter() {
const [value, setValue] = useState(0);
const increment = () => {
setValue(value + 1);
};
const decrement = () => {
setValue(value - 1);
};
return (
<div>
{value}
<button onClick={increment}>+</button>
<button onClick={decrement}>-</button>
</div>
);
}

Should you use this solution with the components that are nearly full of states? Probably not. This is, unless you have a plan to take  advantage of this additional intermediation. These days having a plan is the key.

The Redux library is just a toolkit for  “mounting” reducers to the global object of a single store. You can use as little or as much Redux as you want.

For example with the library like  React Query you can get rid of :

  • Connectors
  • ActionCreators
  • Middleware
  • Reducers
  •  Loading/Error/Result states
  • Contexts

After removing all of these things, you may be asking yourself a question: „Is it worth it to continue to use any client state manager for such a tiny global state?”

In my opinion, Redux library should be the last resort when choosing state management solutions , however the decision is entirely up to You!

Dodaj komentarz