如何正确使用 redux toolkit 的 preloadedState 和 createSlice?

时间:2021-07-29 07:54:49

标签: redux redux-toolkit

我正在尝试迁移到 redux 工具包,但遇到了一个问题。

这是一个简单的计数器切片示例。

import { createSlice } from "@reduxjs/toolkit";

const initialState = {
  value: 0,
};

export const counterSlice = createSlice({
  name: "counter",
  initialState,
  reducers: {
    increment: (state) => {
      state.value += 1;
    },
    decrement: (state) => {
      state.value -= 1;
    },
  },
});

export const { increment, decrement } = counterSlice.actions;
export default counterSlice.reducer;

这是一个用 configureStore 创建的商店。

import { configureStore } from "@reduxjs/toolkit";
import counterReducer from "./slice";

export const store = configureStore({
  reducer: {
    counter: counterReducer,
    // later, many other reducers will be added here.
  },
});

还有什么问题。

但是如果我引入preloadedState,就会出现问题。

const store = configureStore({
    reducer: counterReducer,
    preloadedState: {
      counter: {
         value: 10
      }
    },
  });

如果我像下面这样记录商店的状态,它会按预期记录。

   // without using preloadedState, it results to {counter: {value: 0}}
   console.log(store.getState())
   // with using preloadedState, it results to {counter: {value: 10}}
   console.log(store.getState())

但是在使用 slice reducer 时会出现一个问题,因为 slice reducer 使用自己的状态。

   ...
     reducers: {
       increment: (state) => {
         state.value += 1;  // not work anymore if using preloadedState. we have to use state.counter.value instead.
       },
       decrement: (state) => {
         state.value -= 1;  // not work anymore if using preloadedState. we have to use state.counter.value instead.
       },
     },
   ...

相反,我们必须使用,

   ...
     reducers: {
       increment: (state) => {
         state.counter.value += 1; 
       },
       decrement: (state) => {
         state.counter.value -= 1;
       },
     },
   ...

那么问题来了,我们是否需要添加条件语句,根据我们是否使用preloadedState来分离slice reducer内部的逻辑?

如果切片 reducer 使用 preloadedState 而不是使用它自己的状态,那么无论何时提供 preloadedState 都会很好。有没有更好的方法?

谢谢。

1 个答案:

答案 0 :(得分:2)

如果你查看你的 Redux Devtools 浏览器扩展,你会注意到 - 因为你只有一个 reducer 并且你将它作为根 reducer 传入,你的 state 的形状为

{
  value: 10
},

此时您的正常状态只是没有计数器减速器的子键,因为您的整个状态计数器减速器。

如果你想让你的状态有一个合理的结构(并且在未来能够使用多个reducer),你必须添加reducer键作为一个对象:

const store = configureStore({
    reducer: {
      counter: counterReducer,
    },
    preloadedState: {
      counter: {
         value: 10
      }
    },
  });

正如您在 devtools 中所注意到的,现在您的状态结构是您所期望的 - 并且预加载的状态也将与之匹配。

但请记住,从现在开始,在选择器中您必须useSelector(state => state.counter.value)