In this article, we embark on a journey to share our firsthand experience and insights. We'll illustrate the compelling reasons that drove us to transition from Redux to React Query, emphasizing the advantages it brings to the table. Through a practical example, we'll demonstrate how React Query simplifies and modernizes our state management approach.
Whether you're an experienced Redux user looking for a modern solution or new to state management choices, this article is a guide to help you make an informed decision to enhance your React application's state management.
TL;DR: here are our key benefits of React Query over Redux:
- Simplifies state management with automatic handling of error, loading, and fetching states.
- Reduces code complexity by eliminating the need for manual state management.
- Enhances maintainability and debugging with more readable and straightforward code.
- Avoids complex concepts like reducers, selectors, thunks, and dispatch, making it user-friendly and efficient.

Server State vs. Client State
The choice depends on your application’s requirements, with React Query excelling in server data scenarios, while Redux is robust for client-side state management. If you’re looking for additional resources on this topic, you can read Client vs. Server and replace client-side.
In contrast, Redux focuses on client-side state, centralizing application state management within the Client. Redux is suitable when you need precise control over the client-side state.
The choice depends on your application’s requirements, with React Query excelling in server data scenarios, while Redux is robust for client-side state management. If you’re looking for additional resources on this topic, you can read Client vs. Server and replace client-side.
4 Compelling Advantages of React Query Over Redux
React Query offers several advantages, making it an attractive choice for modernizing our state management:
Simplified Asynchronous Data Handling
Redux often requires writing verbose asynchronous action creators and middleware to manage data fetching and updating. React Query simplifies this process by providing a clean and straightforward way to handle asynchronous data fetching. With React Query, you can easily define queries and mutations, abstracting away much of the complexity that Redux typically involves.
Efficient Data Caching
One of React Query's standout features is its built-in caching mechanism. It intelligently stores and updates data on the client side, reducing the need for redundant network requests. In contrast, Redux relies heavily on manual caching implementations, which can be error-prone and less efficient.
Real-time Data Reactivity
While Redux primarily focuses on managing the application's global state, React Query extends its capabilities by offering real-time data reactivity out of the box. This means that when data changes on the server, React Query can automatically update your UI without the need for additional setup. Redux often requires the integration of libraries like Redux-Saga or Redux-Thunk to achieve similar functionality.
Ecosystem and Community Support
React Query has gained significant momentum in the React community, and it continues to receive active development and support. As a result, you can benefit from an extensive ecosystem of plugins and community-contributed integrations. Redux, while still widely used, has seen a shift in popularity toward more simple and modern alternatives like React Query.
Transitioning from Redux to React Query
We'll start by presenting a simplified Redux implementation, focusing on the key aspects rather than the full architectural details.
Firstly, let's take a look at the slice responsible for fetching User Account Information:
import { createSlice, createAsyncThunk } from '@reduxjs/toolkit'
import { type AccountInfo, AccountInfoApi } from 'qovery-typescript-axios' // Import your API or service for user data here
interface UserState {
data: AccountInfo | null
loading: boolean
error: string | null
}
const accountApi = new AccountInfoApi()
// Create an asynchronous action to perform the request
export const fetchAccountInformation = createAsyncThunk(
'user/fetchAccountInformation',
async () => {
const result = await accountApi.getAccountInformation()
return result
}
)
export const initialUserState: UserState = {
data: null, // Initialize user data to null
loading: false,
error: null,
}
// Create a slice to manage user state
const userSlice = createSlice({
name: 'user',
initialState: initialUserState,
reducers: {},
extraReducers: (builder) => {
builder
.addCase(fetchAccountInformation.pending, (state) => {
state.loading = true;
})
.addCase(fetchAccountInformation.fulfilled, (state, action) => {
state.loading = false;
state.data = action.payload;
})
.addCase(fetchAccountInformation.rejected, (state, action) => {
state.loading = false;
state.error = action.error.message;
});
},
})
export default userSlice.reducer
// Export the action to reuse it in your components
export { fetchAccountInformation }
Now, let's consider a basic usage example within a custom component called UserProfile:
import React, { useEffect } from 'react'
import { useDispatch, useSelector } from 'react-redux'
import { fetchAccountInformation } from './user-slice.ts' // Import the action
function UserProfile() {
const dispatch = useDispatch()
const { data, loading, error } = useSelector((state) => state.user)
useEffect(() => {
// Dispatch the fetchAccountInformation action when the component mounts
dispatch(fetchAccountInformation())
}, [dispatch])
if (loading) {
return
Loading user information...
;
}
if (error) {
return
Error: {error}
;
}
if (!data) {
return
No user data available.
;
}
return (
User Profile
Name: {data.name}
)
}
export default UserProfile
Indeed, the Redux Slice approach can become verbose, and while using useEffect in this context is necessary, it introduces some complexity. However, in React Query, we can achieve the same functionality with remarkable simplicity. Take a look at how the Redux Slice code can be entirely replaced by the following concise React Query example:
export function useUserAccount() { return useQuery({ queryKey: ['userAccount'], queryFn: async () => { const result = await accountApi.getAccountInformation() return result.data; }, }) }
With React Query, we not only eliminate the need for Redux Slice, but we also gain automatic management of loading, error handling, and data retrieval - all in just a few lines of code. This simplicity and efficiency are some of the key advantages that make the transition from Redux to React Query a compelling choice for state management in React applications.
You can use it on our UserProfile removing useEffect and Redux dispatch, selector, by this:
const { data, isError, isLoading } = useUserAccount()
Now, let's explore how to simplify data access with React Query and Query Key Factory while enhancing type safety. Let's get started!
