title | nav |
---|---|
如何从 v4 迁移到 v5 |
23 |
我们强烈建议在迁移到 v5 之前更新到 v4 的最新版本。它会显示所有弃用警告,而不会破坏您的应用程序。
- 删除默认导出
- 删除弃用功能
- 将 React 18 设为最低要求版本
- 将 use-sync-external-store 设为对等依赖项(在
zustand/traditional
中用于createWithEqualityFn
和useStoreWithEqualityFn
) - 将 TypeScript 4.5 设为最低要求版本
- 删除 UMD/SystemJS 支持
- 在 package.json 中组织入口点
- 删除 ES5 支持
- 当 setState 的 replace 标志设置时,更严格的类型
- 持久化中间件行为变化
- 其他小改进(技术上是重大变化)
v5 中的 create
函数不支持自定义相等函数。
如果您使用自定义相等函数如 shallow
,最简单的迁移方法是使用 createWithEqualityFn
。
// v4
import { create } from 'zustand'
import { shallow } from 'zustand/shallow'
const useCountStore = create((set) => ({
count: 0,
text: 'hello',
// ...existing code...
}))
const Component = () => {
const { count, text } = useCountStore(
(state) => ({
count: state.count,
text: state.text,
}),
shallow,
)
// ...existing code...
}
这可以在 v5 中使用 createWithEqualityFn
完成:
npm install use-sync-external-store
// v5
import { createWithEqualityFn as create } from 'zustand/traditional'
// 其余部分与 v4 相同
或者,对于 shallow
用例,您可以使用 useShallow
钩子:
// v5
import { create } from 'zustand'
import { useShallow } from 'zustand/shallow'
const useCountStore = create((set) => ({
count: 0,
text: 'hello',
// ...existing code...
}))
const Component = () => {
const { count, text } = useCountStore(
useShallow((state) => ({
count: state.count,
text: state.text,
})),
)
// ...existing code...
}
v5 中有一个行为变化,以匹配 React 的默认行为。如果选择器返回一个新引用,它可能会导致无限循环。
例如,这可能会导致无限循环。
// v4
const [searchValue, setSearchValue] = useStore((state) => [
state.searchValue,
state.setSearchValue,
])
错误消息将类似于:
Uncaught Error: Maximum update depth exceeded. This can happen when a component repeatedly calls setState inside componentWillUpdate or componentDidUpdate. React limits the number of nested updates to prevent infinite loops.
要修复它,请使用 useShallow
钩子,它将返回一个稳定的引用。
// v5
import { useShallow } from 'zustand/shallow'
const [searchValue, setSearchValue] = useStore(
useShallow((state) => [state.searchValue, state.setSearchValue]),
)
这是另一个可能导致无限循环的示例。
// v4
const action = useMainStore((state) => {
return state.action ?? () => {}
})
要修复它,请确保选择器函数返回一个稳定的引用。
// v5
const FALLBACK_ACTION = () => {}
const action = useMainStore((state) => {
return state.action ?? FALLBACK_ACTION
})
或者,如果您需要 v4 行为,createWithEqualityFn
可以做到。
// v5
import { createWithEqualityFn as create } from 'zustand/traditional'
- setState:
- (partial: T | Partial<T> | ((state: T) => T | Partial<T>), replace?: boolean | undefined) => void;
+ setState:
+ (partial: T | Partial<T> | ((state: T) => T | Partial<T>), replace?: false) => void;
+ (state: T | ((state: T) => T), replace: true) => void;
如果您不使用 replace
标志,则不需要迁移。
如果您使用 replace
标志并将其设置为 true
,则必须提供完整的状态对象。此更改确保 store.setState({}, true)
(导致无效状态)不再被视为有效。
示例:
// 部分状态更新(有效)
store.setState({ key: 'value' })
// 完整状态替换(有效)
store.setState({ key: 'value' }, true)
// 不完整状态替换(无效)
store.setState({}, true) // 错误
如果 replace
标志的值是动态的并在运行时确定,您可能会遇到问题。要处理此问题,您可以通过使用 setState
函数的参数注释 replace
参数来使用变通方法:
const replaceFlag = Math.random() > 0.5
const args = [{ bears: 5 }, replaceFlag] as Parameters<
typeof useBearStore.setState
>
store.setState(...args)
以前,persist
中间件在存储创建期间存储初始状态。此行为在 v5(和 v4.5.5)中已被删除。
例如,在以下代码中,初始状态存储在存储中。
// v4
import { create } from 'zustand'
import { persist } from 'zustand/middleware'
const useCountStore = create(
persist(
() => ({
count: Math.floor(Math.random() * 1000),
}),
{
name: 'count',
},
),
)
在 v5 中,不再是这种情况,您需要在存储创建后显式设置状态。
// v5
import { create } from 'zustand'
import { persist } from 'zustand/middleware'
const useCountStore = create(
persist(
() => ({
count: 0,
}),
{
name: 'count',
},
),
)
useCountStore.setState({
count: Math.floor(Math.random() * 1000),
})