突破Zustand的局限性:与React ContentAPI搭配使用

news/2024/11/28 23:44:36/

Zustand在状态管理中是非常适手的工具,在很多场景中我们都可以用它来解决复杂问题.

但是由于Zustand的设计理念,它仍然有一些限制,在这里用官网中的小demo举一个很简单的例子:

import { create } from 'zustand'
type CountStore = {count: numberinc: () => void
}
export const useCountStore = create<CountStore>((set) => ({count: 1,inc: () => set((state) => ({ count: state.count + 1 })),
}))

如上所示,我们在其他地方可以很灵活的使用count并且简易的控制inc函数count的值做修改。但是如果我们想要为此函数中的count设定灵活初始值的话,或者说是想要依靠React组件中传递来的一些值对其进行初始化的话,我们该怎么做呢?

事实上我们可以使用useEffect来对初值进行修改:

import { useCountStore } from "@/store/countStore"
type AppProps = {initialCount: number
}
export default function App ({initialCount}: AppProps) {count { count } = useCountStore()React.useEffect(() => {useCountStore.setState((prev) => ({...prev, count: initialCount}))}, [])return (<div>{ count }</div>)
}

在这个例子中,useEffect会在组件挂载后调用,并修改 store 中的count值。注意到初次渲染时,count会是初始值1,然后通过useEffect更新为传入的 initialCount。这种方式虽然可以解决问题,但它的缺点是需要两次渲染:第一次渲染时显示默认值,第二次渲染时显示传入的initialCount

这是遇到这个问题的很普遍或是说很简单的解决方法,不过很明显,它并不是一个很好的方法。

这样的处理逻辑是在组件中首先渲染了count的值为1,然后我们在useEffect中调用函数将count的值修改为我们所需的initialCount(更新我们所需的状态),然后组件会触发重新渲染,它在组件中将显示正确的值。这相当于我们用两次渲染来实现同样的事情,这在大型应用中是会极大的影响性能的


这篇文章中要强调的就是另外一种更为优雅的方式:

ZustandReactContextAPI结合使用

仍然是承接上文举一个小demo:

import React, { PropsWithChildren, useState } from "react";
import { StoreApi, createStore } from "zustand";
import { useStore } from "zustand/react";type CountStore = {count: number;inc: () => void;
};const CountContext = React.createContext<StoreApi<CountStore> | undefined>(undefined);type CountProviderProps = React.PropsWithChildren & {initialCount: number; // 设定初始值给 count
};export default function CountProvider({ children, initialCount }: CountProviderProps) {// 创建 store 对象const [store] = useState(() =>createStore<CountStore>((set) => ({count: initialCount, // 使用传入的 initialCount 作为初始值inc: () => set((state) => ({ count: state.count + 1 })),})));return <CountContext.Provider value={store}>{children}</CountContext.Provider>;
}export function useCountStore<T>(selector: (state: CountStore) => T) {const context = React.useContext(CountContext);if (!context) {throw new Error("CountContext.Provider is missing!");}return useStore(context, selector);
}

在上述代码中,在渲染时我们使用useState初始化了Zustandstore,使用传入的initialCount作为初始值。我们利用ContentAPI不会重新渲染的机制,确保在store状态发生改变的时候,CountProvider不会重新渲染,这样很有效的避免了性能上的问题,也解决了我们所谓初始store的“痛点”

最后我们可以按照如下方法进行初始化以及使用

import CountProvider, { useCountStore } from './CountProvider';function App() {return (<CountProvider initialCount={10}><ChildComponent /></CountProvider>);
}function ChildComponent() {const count = useCountStore((state) => state.count);return <div>Count: {count}</div>;
}

虽然通过React Context提供 store 是一种较好的方法,但它依然引入了一定的复杂性,尤其是在大规模应用中。如果有很多状态需要管理或者是组件数非常非常深的时候,不得不承认Context确实会带来一些额外的开销,这样我们就需要根据具体的场景制定特殊的方案或者使用其他相关工具来解决问题

总而言之言而总之,在我们需要状态管理的时候,还是要根据应用的复杂度和性能需求进行权衡,确保我们选取的工具较为合理

灵感来源:TkDodo’s blog / Zustand and React Context


http://www.ppmy.cn/news/1550755.html

相关文章

numpy.digitize函数介绍

numpy.digitize 是 NumPy 中的一个函数,用于根据给定的边界(分箱规则)对输入数组中的值进行分箱,返回对应的箱索引。它在分段处理、统计分布和离散化数据等场景中非常有用。 函数签名 numpy.digitize(x, bins, right=False)参数详解 x: 输入数组,表示需要进行分箱的值。…

PG17 外链设置

很多工具不太支持PG17,看了下要 WINDOS测试链接 查看端口 [sharksharkdb>~]$pg_isready/tmp:5432- accepting connections 查看PG日志 ​​​​​​​ 2024-11-22 05:23:27.361 CST [113921] LOG: invalid authentication method"192.168.2.0/24"2024-11-22 0…

Qt程序发布及打包成exe安装包

参考:Qt之程序发布以及打包成exe安装包 目录 一、简述 Qt 项目开发完成之后,需要打包发布程序,而因为用户电脑上没有 Qt 配置环境,所以需要将 release 生成的 exe 文件和所依赖的 dll 文件复制到一个文件夹中,然后再用 Inno Setup 打包工具打包成一个 exe 安装包,就可以…

云计算的计算包括哪些内容

‌云计算的计算主要包括以下几种类型‌&#xff1a; ‌分布式计算‌&#xff1a;分布式计算是一种计算方法&#xff0c;它将大型问题分解成多个小任务&#xff0c;然后分配给多个计算机进行处理。这种方法可以提高计算效率和可靠性‌1。‌并行计算‌&#xff1a;并行计算是同时…

map和redis关系

Map 和 Redis 都是用于存储和管理数据的工具&#xff0c;但它们在用途、实现和应用场景上有所不同。下面详细解释 Map 和 Redis 之间的关系和区别。 1. Map 数据结构 定义 Map 是一种数据结构&#xff0c;用于存储键值对&#xff08;key-value pairs&#xff09;。每个键都是…

【逐行注释】自适应观测协方差R的AUKF(自适应无迹卡尔曼滤波,MATLAB语言编写),附下载链接

文章目录 自适应R的UKF逐行注释的说明运行结果部分代码各模块解释 自适应R的UKF 自适应无迹卡尔曼滤波&#xff08;Adaptive Unscented Kalman Filter&#xff0c;AUKF&#xff09;是一种用于状态估计的滤波算法。它是基于无迹卡尔曼滤波&#xff08;Unscented Kalman Filter&…

C# 属性 学习理解记录

字段和属性 左边字段&#xff0c;右边属性 拓展&#xff0c;属性安全&#xff1a; 1、设置public private 和protected 等&#xff0c;只读&#xff0c;只写&#xff0c; 2、在get set 方法时&#xff0c;验证&#xff0c;异常时抛出错误

【数据结构笔记】习题

渐进分析 【2010-THU-Mid】f(n) O(g(n))&#xff0c;当且仅当g(n) Ω(f(n))。&#xff08;√&#xff09; 【2010-THU-Mid】若f(n) O(n^2)且g(n) O(n)&#xff0c;则以下结论正确的是&#xff08;AD&#xff09; A. f(n) g(n) O(n^2) B. f(n) / g(n) O(n) C. g(n) O(f(…