React Query + Redux toolkit 封装异步请求

当你需要进行 Redux 和 React Query 的组合时,除了常规的 Redux 方法(例如手动派发 action 和更新 state),还可以使用 createSlice 和 React Query 进行组合,这可以让你更方便地封装异步请求和更容易地更新状态。

使用 createSlice 创建 reducer:

javascript 复制代码
import { createSlice } from '@reduxjs/toolkit';
import { fetchUserInfo, fetchUserList } from './api';
import { useQuery } from 'react-query';

const userSlice = createSlice({
  name: 'users',
  initialState: {
    userInfo: null,
    userList: null,
  },
  reducers: {
    setUserInfo: (state, action) => {
      state.userInfo = action.payload;
    },
    setUserList: (state, action) => {
      state.userList = action.payload;
    },
  },
});

export const { setUserInfo, setUserList } = userSlice.actions;

export const useUserInfoQuery = (userId) => {
  const { data, isLoading, isError } = useQuery(['userInfo', userId], () => fetchUserInfo(userId));

  if (data) {
    dispatch(setUserInfo(data));
  }

  return {
    user: useSelector((state) => state.users.userInfo),
    isLoading,
    isError,
  };
};

export const useUserListQuery = () => {
  const { data, isLoading, isError } = useQuery('userList', fetchUserList);

  if (data) {
    dispatch(setUserList(data));
  }

  return {
    userList: useSelector((state) => state.users.userList),
    isLoading,
    isError,
  };
};

export default userSlice.reducer;

接着,在 store 中注册 userReducer reducer,并添加 React Query Provider,这样可以在整个应用程序中共享 React Query 的 cache,避免重复的请求。

javascript 复制代码
import { configureStore } from '@reduxjs/toolkit';
import { QueryClient, QueryClientProvider } from 'react-query';
import { Provider } from 'react-redux';
import userReducer from './userSlice';

const queryClient = new QueryClient();

const store = configureStore({
  reducer: {
    users: userReducer,
  },
});

ReactDOM.render(
  <Provider store={store}>
    <QueryClientProvider client={queryClient}>
      <App />
    </QueryClientProvider>
  </Provider>,
  document.getElementById('root')
);

最后,在组件中使用创建的 useUserInfoQueryuseUserListQuery hooks,展示用户信息和用户列表。

javascript 复制代码
import { useUserInfoQuery, useUserListQuery } from './userSlice';

const UserInfo = ({ userId }) => {
  const { user, isLoading, isError } = useUserInfoQuery(userId);

  if (isLoading) {
    return <div>Loading</div>;
  }

  if (isError) {
    return <div>Error</div>;
  }

  return (
    <div>
      <h1>{user.name}</h1>
      <p>{user.email}</p>
      <p>{user.phone}</p>
    </div>
  );
};

const UserList = () => {
  const { userList, isLoading, isError } = useUserListQuery();

  if (isLoading) {
    return <div>Loading</div>;
  }

  if (isError) {
    return <div>Error</div>;
  }

  return (
    <div>
      {userList.map(user => (
        <div key={user.id}>
          <h2>{user.name}</h2>
          <p>{user.email}</p>
          <p>{user.phone}</p>
        </div>
      ))}
    </div>
  );
};

通过使用 createSlice 和 React Query 结合的方式,不仅可以更轻松管理 Redux 的异步状态和数据请求,也更方便地管理用户的相关状态信息。

相关推荐
pe7er4 小时前
状态提升:前端开发中的状态管理的设计思想
前端·vue.js·react.js
晚风予星6 小时前
Ant Design Token Lens 迎来了全面升级!支持在 .tsx 或 .ts 文件中直接使用 Design Token
前端·react.js·visual studio code
程序猿的程8 小时前
开源一个 React 股票 K 线图组件,传个股票代码就能画图
前端·javascript
大雨还洅下9 小时前
前端JS: 虚拟dom是什么? 原理? 优缺点?
javascript
唐叔在学习9 小时前
[前端特效] 左滑显示按钮的实现介绍
前端·javascript
青青家的小灰灰10 小时前
React 架构进阶:自定义 Hooks 的高级设计模式与最佳实践
前端·react.js·前端框架
青青家的小灰灰10 小时前
深入理解事件循环:异步编程的基石
前端·javascript·面试
前端Hardy11 小时前
HTML&CSS&JS:打造丝滑的3D彩纸飘落特效
前端·javascript·css
前端Hardy11 小时前
HTML&CSS&JS:丝滑无卡顿的明暗主题切换
javascript·css·html
UIUV12 小时前
node:child_process spawn 模块学习笔记
javascript·后端·node.js