开发者问题收集

Redux 初始状态有值,但在 redux dev-tool 中显示为 null。为什么?

2023-11-20
261

我遇到了此错误,我已为 redux 的 initialState 设置了一个值,但在调试时它显示其值为空。我正在使用 Redux Dev-tools 来检查这些值。 此外,在 initialState 中添加数据时,它会引发错误 Uncaught TypeError: Cannot read properties of null (reading 'push')

这是我的代码:

Store.js

import { configureStore } from "@reduxjs/toolkit";
import { persistStore, persistReducer } from "redux-persist";
import storage from "redux-persist/lib/storage";
import authReducer from "../feature/auth/auth.js";

const persistConfig = {
  key: "root",
  storage,
};

const persistedReducer = persistReducer(persistConfig, authReducer);

export const store = configureStore({
  reducer: {
    auth: persistedReducer, // Use the persisted reducer here
  },
});

export const persistor = persistStore(store);

Feature.js

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

const initialState = {
  cookies: [{value: 'test'}]
};

const authSlice = createSlice({
  name: "auth",
  initialState,
  reducers: {
    setCookies: (state, action) => {
      const data = {
        value: action.payload.value
      }
      state.cookies.push(data)
    },
    removeCookies: (state) => {
      state.cookies = [];
    }
  },
});

export const { setCookies, removeCookies } = authSlice.actions;
export default authSlice.reducer;

这是我尝试添加 cookie 的方式:

import { useDispatch } from "react-redux";
import { setCookies } from "../feature/auth/auth";

function About() {
  const dispatch = useDispatch()

  const handleCookies = () => {
    dispatch(setCookies({value: 'test'}))
  }

  return (
    <div>
      <h1>About</h1>
      <p>This is the about page</p>
      <button onClick={handleCookies}>Add Cookeis</button>
    </div>
  )
}

export default About;

Redux

2个回答

看来您已保留了某些先前的状态,其中 state.cookies 空值。当应用加载时,Redux 存储将重新水化,并且持久的空值将恢复回状态。

当状态形状发生如下变化时,您有几个选项:

  1. 从 localStoreage 中删除持久状态并重新加载页面。这允许应用挂载/加载,并且由于没有持久状态来重新水化存储,因此 Reducer 将使用其初始状态值,例如 authSlice 将初始化为 { cookies: [{value: 'test'}]

  2. 将旧状态形状迁移到新状态形状。有关详细信息,请参阅 Redux-Persist Migrations

    基本示例:

    import { configureStore } from "@reduxjs/toolkit";
    import { createMigrate, persistReducer, persistStore } from 'redux-persist';
    import storage from "redux-persist/lib/storage";
    import authReducer from "../feature/auth/auth.js";
    
    const migrations = {
    // 将具有无效 cookies 值的状态迁移到数组中
    0: (state) => {
    return {
    ...state,
    auth: {
    ...state.auth,
    cookies: [],
    },
    };
    },
    };
    
    const persistConfig = {
    key: "root",
    version: 0, // <-- 持久状态版本
    storage,
    move: createMigrate(migrations, { debug: false }), // <-- 迁移状态
    };
    
    const persistedReducer = persistReducer(persistConfig, authReducer);
    
    export const store = configureStore({
    reducer: {
    auth: persistedReducer,
    },
    });
    
    export const persistor = persistStore(store);
    

如果您仍然只是在进行本地开发,并且您的应用没有处于无效状态的版本,那么选项 1 就可以了,因为它实际上只会影响您。但是,一旦有用户在使用您的应用,您将需要使用选项 2,以便可以优雅地迁移他们的状态版本。

Drew Reese
2023-11-20

由于您的本地存储具有 {cookies: null ,因此 cookies 似乎为空。可能是来自开发早期的代码?

Merging involves saving the persisted state back in the Redux store. When our app launches, our initial state is set. Shortly after, Redux Persist retrieves our persisted state from storage, then overrides any initial state. This process works automatically.

https://blog.logrocket.com/persist-state-redux-persist-redux-toolkit-react/

一种可能的解决方案是删除您的本地存储,允许 cookies 值正确初始化为数组,而不是 null。这还可以防止抛出 Uncaught TypeError: Cannot read properties of null (reading 'push') ,因为它将不再对 null 调用 .push() 。>

Chris Warren
2023-11-20