react18中使用redux管理公共数据仓库实现数据immutable更新

Immutable.js出自Facebook,是最流行的不可变数据结构的实现之一。它实现了完全的持久化数据结构,使用结构共享。所有的更新操作都会返回新的值,但是在内部结构是共享的,来减少内存占用。Immutablejs官网

在上一篇介绍redux的文章,我们可以看到在创建的reducer中进行数据更改的时候,会使用...来复制状态数据,为的就是不对原数据进行直接修改,只做替换,就是为了实现react的数据不可变immutable。也就是说如果state的数据比较简单那还好,万一比较复杂,就要复制很多这样的结构数据,很占内存影响app的性能。

安装

bash 复制代码
npm i immutable redux-immutable -S

替换redux中的combineReducers,改用redux-immutable里面的

js 复制代码
import { createStore } from "redux";
import { combineReducers } from "redux-immutable";

import { CounterReducer } from "./CounterReducer";

const reducers = combineReducers({
  count: CounterReducer,
});

let store = createStore(reducers);

store.subscribe(() => console.log(store.getState()));

export default store;

改造reducers里面的代码

js 复制代码
import { fromJS } from "immutable";
const initialState = fromJS({
  counter: 0,
  userInfo: {
    name: "John Doe",
    age: 25,
  },
});
function CounterReducer(state = initialState, action) {
  switch (action.type) {
    case "ADD":
      return state.update("counter", (val) => {
        console.log("🚀 ~ returnstate.update ~ val:", val);
        return val + 1;
      });
    case "DEC":
      return state.update('counter',value => value - 1)
    default:
      return state;
  }
}

export { CounterReducer };

组件里面使用方式变化

DemoA.js

js 复制代码
import store from "../../store";
function DemoA() {
  const count = store.getState().get("count").get("counter");
  return (
    <div>
      <p> Demo A count: {count}</p>
    </div>
  );
}
export default DemoA;

DemoB.js

js 复制代码
import store from "../../store";

function DemoB() {
  const data = store.getState().get("count");
  return (
    <div>
      <p>Demo B</p>
      <div>
        <span>name: {data.getIn(["userInfo", "name"])}</span> <br />
        <span>age: {data.getIn(["userInfo", "age"])}</span>
        <span>count:{data.get("counter")}</span>
      </div>
      <button onClick={() => store.dispatch({ type: "ADD" })}>add count</button>
    </div>
  );
}
export default DemoB;

index.js

js 复制代码
import DemoA from "./DemoA";
import DemoB from "./DemoB";
import store from "../../store/index";
import { useEffect, useState } from "react";
function Redux() {
  console.log(store.getState(), "store.getState()");
  const [counter, setCount] = useState(
    store.getState().get("count").get("counter")
  );
  useEffect(() => {
    const unsubscribe = store.subscribe(() => {
      console.log("aaaa");
      setCount(store.getState().get("count").get("counter"));
    });
    return () => {
      unsubscribe();
    };
  }, [counter]);
  return (
    <div>
      Redux
      <hr />
      parent count:{counter}
      <DemoA />
      <DemoB />
    </div>
  );
}

export default Redux;

实现了同样的效果

相关推荐
haorooms2 分钟前
Promise.try () 完全指南
前端·javascript
kyriewen3 分钟前
闭包:那个“赖着不走”的家伙,到底有什么用?
前端·javascript·ecmascript 6
斌味代码6 分钟前
el-popover跳转页面不隐藏,el-popover销毁
前端·javascript·vue.js
该怎么办呢7 分钟前
cesium核心代码学习-01项目目录及其基本作用
前端·3d·源码·webgl·cesium·webgis
踩着两条虫14 分钟前
AI 驱动的 Vue3 应用开发平台 深入探究(十九):CLI与工具链之Create VTJ CLI 参考
前端·ai编程·vite
天下无贼!25 分钟前
【Python】2026版——FastAPI 框架快速搭建后端服务
开发语言·前端·后端·python·aigc·fastapi
GISer_Jing1 小时前
两种AI交互方式深度解析——浏览器书签&插件
前端·人工智能·ai·prompt
哈__1 小时前
ReactNative项目OpenHarmony三方库集成实战:react-native-device-info
javascript·react native·react.js
前端布鲁伊1 小时前
零代码上线一个图片处理网站,我是如何使唤AI干活的?
前端·ai编程