2023-谈谈复杂应用的状态管理:为什么是Zustand

原文地址

谈谈复杂应用的状态管理:为什么是Zustand

作为一名主业做设计,业余搞前端的小菜鸡,到2020年底为止都是用云谦大佬的dva一把梭。当时整体的使用体验还是挺好的,对于我这样的前端菜鸡上手门槛低,而且学一次哪都可用,当时从来没愁过状态管理。

img

至今Kitchen3里仍然躺着用dva做状态管理的功能模块,写于2020

直到hooks横空出世, TypeScript逐步流行。一方面,从react hooks出来以后,大量的文章开始鼓吹「你不需要Redux「useState + Context」完全可用「next-unstated」YYDS等等。另一方面,由于Dva不再维护,其在ts下的都没有任何提示的问题也逐步暴露。

在尝试一些小项目中使用hooks后感觉还行之后,作为小萌新的我也全面转向了hooks的怀抱。中间其实一直没怎么遇到问题,因为大部分前端应用的复杂度也就那样,hooks问题不大。然后呢?然后从去年开始就在复杂应用里踩坑了。

img

复杂应用的状态管理天坑

ProEditor是内部组件库TechUI Studio的编辑器组件。

业务组件ProEditor就是一个很典型的例子。由于ProEditor是个编辑器,对用户来说编辑体验非常重要,是一个重交互操作的应用,这就会牵扯到大量的状态管理需求。

img

先简单来列下ProEditor的状态管理需求有哪些:

Editor容器状态管理与组件(Table)状态管理拆分,但可联动消费;

容器状态负责了一些偏全局配置的状态维护,比如画布、代码页的切换,是否激活画布交互等等,而组件的状态则是保存了组件本身的所有配置和状态。

这么做的好处在于不同组件可能会有不同的状态,而Editor的容器状态可以复用,比如做ProForm的时候,Editor的容器仍然可以是同一个,组件状态只需额外实现ProFormStore即可。

img

从上图可以看到,Table的状态就是Editorconfig字段,当Table改时,会触发Editorconfig字段同步更新。当Editor更新时,也会触发该数据更新。

最初的版本,我使用了Provider + Context的方式来做全局状态管理。大概的写法是这样的:

// 定义
export const useStudioStore = (props?: ProEditorProps) => {
  // ...
  const tableStore = useTableStore(props?.value);
  const [tabKey, switchTab] = useState(TabKey.canvas);
  const [activeConfigTab, switchConfigTab] = useState<TableConfigGroup>(TableConfigGroup.Table);
  // ...
}

export const StudioStore = createContextStore(useStudioStore, {});

// 消费
const NavBar: FC<NavBarProps> = ({ logo }) => {
  const { tabKey } = useContext(StudioStore);
  return ...
}

由于这一版是Context一杆推到底,这造成了一些很离谱的交互反馈,就是每一次点击其他任何地方(例如画布代码、组件的配置项,都会造成面板的Tabs重新渲染(左下图。右下图是相应的重渲染分析图,可以看到任何动作都造成了重新所有页面元素的重渲染。而这还是最早期的demo版本,功能和数据量的才实现到20%左右。所以可以预见到如果不做任何优化,使用体验会差到什么程度。

动图封面

❷ 需要进行复杂的数据处理

ProEditor针对表格编辑,做了大量的数据变换操作。比如ProTable中针对 columns 这个字段的更新就有14种操作。比如其中一个比较容易被感知的updateColumnByOneAPI 就是基于oneAPI的字段信息更新,细颗粒度地调整columns里的字段信息。而这样的字段修改类型的store,在ProEditor中除了 columns 还有一个 data

当时,为了保证数据变更方法的可维护性与action的不变性,我采用了userReducer做变更方法的管理。

img

因为一旦采用自定义hooks ,就得写成下面这样才能保证不会重复渲染,会造成极大的心智负担,一旦出现数据不对的情况,很难排查到底是哪个方法或者依赖有问题。

// 自定 hook 的写法
const useDataColumns = () => {
  const createOrUpdateColumnsByMockData = useCallback(() => {
    // ...
  }, [a, b]);
  const createColumnsByOneAPI = useCallback(() => {
    // ...
  }, [c, d]);
  const updateColumnsByOneAPI = useCallback(() => {
    // ...
  }, [a, b, c, d]);
  // ...
};

useReducer也有很大的局限性,例如不支持异步函数、不支持内部的reducer互相调用,不支持和其他state联动(比如要当参数穿进去才可用,所以也不是最优解。

❸ 是个可被外部消费的组件

一旦提到组件,势必要提非受控模式和受控模式。为了支持好我们自己的场景,且希望把ProEditor变成一个好用的业务组件,所以我们做了受控模式,毕竟一个好用的组件一定是要能同时支持好这两种模式的。

在实际场景下,我们既需要配置项(config)受控,同时也需要画布交互状态(interaction)受控,例如下面的场景:在激活某个单元格状态时点击生成,我们需要将这个选中状态进行重置,才能生成符合预期的设计稿。

动图封面

所以为了支持细颗粒度的受控能力,我们提供了多个受控值,供外部受控模式。

// ProEditor 外部消费的 Demo 示意
export default () => {
  const [status, setStatus] = useState();
  const { config, getState } = useState();

  return (
    <ProEditor
      // configonConfigChange 是一对
      config={config}
      onConfigChange={({ config }) => {
        setConfig(config);
      }}
      // interaction 和 onInteractionChange 是另一对受控
      interaction={status}
      onInteractionChange={(s) => {
        setStatus(s);
      }}
    />
  );
};

但当我们一开始写好这个受控api,得到结果是这样的:

动图

对,你没看错,死循环了。 遇到这个问题时让人头极度秃,因为原本以为是个很简单的功能,但是在React生命周期里的表现让人费解,尤其是使用useEffect做状态管理的时候。

// 导致死循环的写法
const useTableStore = (state: Partial<Omit<ProTableConfigStore, 'columns' | 'data'>>) => {
  const { defaultConfig, config: outsourceValue, mode } = props;
  const { columns, isEmptyColumns, dispatchColumns } = useColumnStore(defaultConfig?.columns, mode);
  // 受控模式 内部值与外部双向通信
  useEffect(() => {
    // 没有外部值和变更时不更改
    if (!outsourceValue) return;
    // 相等值的时候不做更新
    if (isEqual(dataStore, outsourceValue)) return;
    if (outsourceValue.columns) {
      dispatchColumns({ type: 'setAll', columns: outsourceValue.columns });
    }
  }, [dataStore, outsourceValue]);

  const dataStore = useMemo(() => {
    const v = { ...store, data, columns } as ProTableConfigStore;
    // dataStore 变更时需要对外变更一次
    if (props.onChange && !isEqual(v, outsourceValue)) {
      props.onChange?.({
        config: v,
        props: tableAsset.generateProps(v),
        isEmptyColumns,
      });
    }
    return v;
  }, [data, store, columns, outsourceValue]);

  // ...
}

造成上述问题的原因大部分都是因为组件内onChange的时机设置。一旦代码里用useEffect的方式去监听变更触发onChange,有很大的概率会造成死循环。

❹ 未来还希望能支持撤销重做、快捷键等能力

毕竟,现代的编辑器都是支持快捷键、历史记录、多人协同等增强型的功能的。这些能力怎么在编辑器的状态管理中以低成本、易维护的方式进行实施,也非常重要。

img

总之,开发ProEditor的经历,一句话的血泪教训就是:

复杂应用的状态管理真的不能裸写hooks

那些鼓吹裸写hooks的人大概率是没遇到过复杂case,性能优化、受控、action互调、数据切片、状态调试等坑,每一项都不是好惹的主,够人喝上一壶。

img

为什么是Zustand

其实,复杂应用只是开发者状态管理需求的集中体现。如果我们把状态管理当成一款产品来设计,我们不妨看看开发者在状态管理下的核心需求是什么。

我相信通过以下这一串分析,你会发现zustand是真真正正满足「几乎所有」状态管理需求的工具,并且在很多细节上做到了体验更优。

img

❶ 状态共享

状态管理最必要的一点就是状态共享。这也是context出来以后,大部分文章说不需要redux的根本原因。因为context可以实现最最基础的状态共享。但这种方法(包括redux在内,都需要在最外层包一个ProviderContext中的值都在Provider的作用域下有效。

// Context 状态共享

// store.ts
export const StoreContext = createStoreContext(() => { ... });

// index.tsx
import { appState, StoreContext } from './store';

root.render(
    <StoreContext.Provider value={appState}>
      <App />
    </StoreContext.Provider>
);

// icon.tsx
import { StoreContext } from './store';

const ReplaceGuide: FC = () => {
  const { i18n, hideGuide, settings } = useContext(StoreContext);

  // ...
  return ...
}

zustand做到的第一点创新就是:默认不需要Provider。直接声明一个hooks式的useStore后就可以在不同组件中进行调用。它们的状态会直接共享,简单而美好。

// Zustand 状态共享

// store.ts
import create from "zustand";

export const useStore = create((set) => ({
  count: 1,
  inc: () => set((state) => ({ count: state.count + 1 })),
}));

// Control.tsx
import { useStore } from "./store";

function Control() {
  return (
    <button
      onClick={() => {
        useStore.setState((s) => ({ ...s, count: s.count - 5 }));
      }}
    >
      5
    </button>
  );
}

// AnotherControl.tsx
import { useStore } from "./store";

function AnotherControl() {
  const inc = useStore((state) => state.inc);
  return <button onClick={inc}> +1 </button>;
}

// Counter.tsx
import { useStore } from "./store";

function Counter() {
  const { count } = useStore();
  return <h1>{count}</h1>;
}

由于没有Provider的存在,所以声明的useStore默认都是单实例,如果需要多实例的话,zustand也提供了对应的Provider书写方式,这种方式在组件库中比较常用。 ProEditor也是用的这种方式做到了多实例。

此外,zustandstore状态既可以在react世界中消费,也可以在react世界外消费。

❷ 状态变更

状态管理除了状态共享外,另外第二个极其必要的能力就是状态变更。在复杂的场景下,我们往往需要自行组织相应的状态变更方法,不然不好维护。这也是考验一个状态管理库好不好用的一个必要指标。

hookssetState 是原子级的变更状态,hold不住复杂逻辑;而 useReducerhooks借鉴了redux的思想,提供了dispatch变更的方式,但和reduxreducer一样,这种方式没法处理异步,且没法互相调用,一旦遇上就容易捉襟见肘。

至于redux ,哪怕是最新的 redux-toolkit 中优化大量redux的模板代码,针对同步异步方法的书写仍然让人心生畏惧。

// redux-toolkit 的用法

import { createAsyncThunk, createSlice } from "@reduxjs/toolkit";
import { userAPI } from "./userAPI";

// 1. 创建异步函数
const fetchUserById = createAsyncThunk(
  "users/fetchByIdStatus",
  async (userId, thunkAPI) => {
    const response = await userAPI.fetchById(userId);
    return response.data;
  }
);

const usersSlice = createSlice({
  name: "users",
  initialState: { entities: [], loading: "idle" },
  // 同步的 reducer 方法
  reducers: {},
  // 异步的 AsyncThunk 方法
  extraReducers: (builder) => {
    // 2. 将异步函数添加到 Slice 中
    builder.addCase(fetchUserById.fulfilled, (state, action) => {
      state.entities.push(action.payload);
    });
  },
});

// 3. 调用异步方法
dispatch(fetchUserById(123));

而在zustand中,函数可以直接写,完全不用区分同步或者异步,一下子把区分同步异步的心智负担降到了0

// zustand store 写法

// store.ts
import create from "zustand";

const initialState = {
  // ...
};

export const useStore = create((set, get) => ({
  ...initialState,
  createNewDesignSystem: async () => {
    const { params, toggleLoading } = get();

    toggleLoading();
    const res = await dispatch("/hitu/remote/create-new-ds", params);
    toggleLoading();

    if (!res) return;

    set({ created: true, designId: res.id });
  },
  toggleLoading: () => {
    set({ loading: !get().loading });
  },
}));

// CreateForm.tsx
import { useStore } from "./store";

const CreateForm: FC = () => {
  const { createNewDesignSystem } = useStore();

  // ...
};

另外一个让人非常舒心的点在于,zustand会默认将所有的函数保持同一引用。所以用zustand写的方法,默认都不会造成额外的重复渲染(PS:这里再顺带吹一下WebStorm对于函数和变量的识别能力,非常好用)

img

在下图可以看到,所有zustanduseStore出来的值或者方法,都是橙色的变量,具有稳定引用,不会造成不必要的重复渲染。

img

而状态变更函数的最后一个很重要,但往往又会被忽略的一点,就是方法需要调用当前快照下的值或方法

在常规的开发心智中,我们往往会在异步方法中直接调用当前快照的值来发起请求,或使用同步方法进行状态变更,这会有极好的状态内聚性。

比如说,我们有一个方法叫「废弃草稿,需要获取当前的一个id ,向服务器发起请求做数据变更,同时为了保证当前界面的数据显示有效性,变更完毕后,我们需要重新获取数据。

我们来看看hooks版本和zustand的写法对比,如下所示:

// hooks 版本

export const useStore = () => {
  const [designId, setDesignId] = useState();
  const [loading, setLoading] = useState(false);

  const refetch = useCallback(() => {
    if (designId) {
      mutateKitchenSWR('/hitu/remote/ds/versions', designId);
    }
  }, [designId]);

  const deprecateDraft = useCallback(async () => {
    setLoading(true);
    const res = await dispatch('/hitu/remote/ds/deprecate-draft', designId);
    setLoading(false);

    if (res) {
      message.success('草稿删除成功');
    }

    // 重新获取一遍数据
    refetch();
  }, [designId, refetch]);

  return {
    designId,
    setDesignId,
    loading,
    deprecateDraft,
    refetch,
  }
};
// zustand 写法

const initialState = { designId: undefined, loading: false };

export const useStore = create((set, get) => ({
  ...initialState,
  deprecateDraft: async () => {
    set({ loading: true });
    const res = await dispatch('/hitu/remote/ds/deprecate-draft', get().designId);
    set({ loading: false });

    if (res) {
      message.success('草稿删除成功');
    }

    // 重新获取一遍数据
    get().refetch();
  },
  refetch: () => {
    if (get().designId) {
      mutateKitchenSWR('/hitu/remote/ds/versions', get().designId);
    }
  },
})

可以明显看到,光是从代码量上zustandstorehooks减少了30% 。不过另外容易被大家忽略,但其实更重要的是, hooks版本中互调带来了引用变更的问题

由于 deprecateDraftrefetch 都调用了 designId,这就会使得当 designId 发生变更时,deprecateDraftrefetch 的引用会发生变更,致使react触发刷新。而这在有性能优化需求的场景下非常阴间,会让不该渲染的组件重新渲染。那这也是为什么react要搞一个 useEvent 的原因(RFC

zustand则把这个问题解掉了。由于zustandcreate方法中提供了 get 对象,使得我们可以用get方法直接拿到当前store中最新的state快照。这样一来,变更函数的引用始终不变,而函数本身却一直可以拿到最新的值。

在这一趴,最后一点要夸zustand的是,它可以直接集成useReducer的模式,而且直接在官网提供了示例。这样就意味着之前在ProEditor中的那么多action可以极低成本完成迁移。

// columns 的 reducer 迁移

import { columnsConfigReducer } from './columns';

const createStore = create((set,get)=>({
  /**
   * 控制 Columns 的复杂数据变更方法
   */
  dispatchColumns: (payload) => {
    const { columns, internalUpdateTableConfig, updateDataByColumns } = get();
    // 旧的 useReducer 直接复用过来
    const nextColumns = columnsConfigReducer(columns, payload);

    internalUpdateTableConfig({ columns: nextColumns }, 'Columns 配置');

    updateDataByColumns(nextColumns);
  },
})

❸ 状态派生

状态派生是状态管理中一个不被那么多人提起,但是在实际场景中被大量使用的东西,只是大家没有意识到,这理应也是状态管理的一环。

状态派生可以很简单,也可以非常复杂。简单的例子,比如基于一个name 字段,拼接出对应的url

img

复杂的例子,比如基于rgbhsl值和色彩模式,得到一个包含色彩空间的对象。

img

如果不考虑优化,其实都可以写一个中间的函数作为派生方法,但作为状态管理的一环,我们必须要考虑相应的优化。

hooks场景下,状态派生的方法可以使用 useMemo,例如:

// hooks 写法

const App = () => {
  const [name, setName] = useState("");
  const url = useMemo(() => URL_HITU_DS_BASE(name || ""), [name]);
  // ...
};

zustand用了类似redux selector的方法,实现相应的状态派生,这个方式使得useStore的用法变得极其灵活和实用。而这种selector的方式使得zustand下细颗粒度的性能优化变为可能,且优化成本很低。

// zustand 的 selector 用法

// 写法1
const App = () => {
  const url = useStore((s) => URL_HITU_DS_BASE(s.name || ""));
  // ...
};

// 写法2 将 selector 单独抽为函数
export const dsUrlSelector = (s) => URL_HITU_DS_BASE(s.name || "");
const App = () => {
  const url = useStore(dsUrlSelector);
  // ...
};

由于写法2可以将selector抽为独立函数,那么我们就可以将其拆分到独立文件来管理派生状态。由于这些selector都是纯函数,所以能轻松实现测试覆盖。

img

❹ 性能优化

讲完状态派生后把zustandselector能力后,直接很顺地就能来讲讲zustand的性能优化了。

在裸hooks的状态管理下,要做性能优化得专门起一个专项来分析与实施。但基于zustanduseStoreselector用法,我们可以实现低成本、渐进式的性能优化。

比如ProEditor中一个叫 TableConfig 的面板组件,对应的左下图中圈起来的部分。而右下图则是相应的代码,可以看到这个组件从 useStore 中 解构了 tabKeyinternalSetState 的方法。

img

然后我们用 useWhyDidYouUpdate 来检查下,如果直接用解构引入,会造成什么样的情况:

动图封面

在上图中可以看到,虽然 tabsinternalSetState 没有变化,但是其中的config数据项(data、columns等)发生了变化,进而使得 TableConfig 组件触发重渲染。

而我们的性能优化方法也很简单,只要利用zustandselector,将得到的对象聚焦到我们需要的对象,只监听这几个对象的变化即可。

// 性能优化方法

import shallow from "zustand/shallow"; // zustand 提供的内置浅比较方法
import { useStore, ProTableStore } from "./store";

const selector = (s: ProTableStore) => ({
  tabKey: s.tabKey,
  internalSetState: s.internalSetState,
});

const TableConfig: FC = () => {
  const { tabKey, internalSetState } = useStore(selector, shallow);
};

这样一来,TableConfig的性能优化就做好了~

动图封面

基于这种模式,性能优化就会变成极其简单无脑的操作,而且对于前期的功能实现的侵入性极小,代码的后续可维护性极高。

img

剩下的时间就可以和小伙伴去吹咱优雅的性能优化技巧了~( ̄︶ ̄)↗

img

就我个人的感受上, zustand使用selector来作为性能优化的思路真的很精巧,就像是给函数式的数据流加上了一点点主观意愿上的响应式能力,堪称优雅。

img

❺ 数据分形与状态组合

如果子组件能够以同样的结构,作为一个应用使用,这样的结构就是分形架构。

数据分形在状态管理里我觉得是个比较高级的概念。但从应用上来说很简单,就是更容易拆分并组织代码,而且具有更加灵活的使用方式,如下所示是拆分代码的方式。但这种方式其实我还没大使用,所以不多展开了。

// 来自官方文档的示例

// https://github.com/pmndrs/zustand/blob/main/docs/typescript.md#slices-pattern

import create, { StateCreator } from 'zustand'

interface BearSlice {
  bears: number
  addBear: () => void
  eatFish: () => void
}
const createBearSlice: StateCreator<
  BearSlice & FishSlice,
  [],
  [],
  BearSlice
> = (set) => ({
  bears: 0,
  addBear: () => set((state) => ({ bears: state.bears + 1 })),
  eatFish: () => set((state) => ({ fishes: state.fishes - 1 })),
})

interface FishSlice {
  fishes: number
  addFish: () => void
}
const createFishSlice: StateCreator<
  BearSlice & FishSlice,
  [],
  [],
  FishSlice
> = (set) => ({
  fishes: 0,
  addFish: () => set((state) => ({ fishes: state.fishes + 1 })),
})

const useBoundStore = create<BearSlice & FishSlice>()((...a) => ({
  ...createBearSlice(...a),
  ...createFishSlice(...a),
}))

我用的更多的是基于这种分形架构下的各种中间件。由于这种分形架构,状态就具有了很灵活的组合性,例如将当前状态直接缓存到localStorage。在zustand的架构下, 不用额外改造,直接加个 persist 中间件就好。

// 使用自带的 Persist Middleware
import create from 'zustand'
import {  persist } from 'zustand/middleware'

interface BearState {
  bears: number
  increase: (by: number) => void
}

const useBearStore = create<BearState>(
  persist((set) => ({
    bears: 0,
    increase: (by) => set((state) => ({ bears: state.bears + by })),
  }))
);

ProEditor中,我使用最多的就是 devtools 这个中间件。这个中间件具有的功能就是:将这个StoreRedux Devtools绑定。

// devtools 中间件

// store 逻辑
const vanillaStore = (set, get) => ({
  syncOutSource: (nextState) => {
    set(
      { ...get(), ...nextState },
      false,
      `受控更新:${Object.keys(nextState).join(" ")}`
    );
  },
  syncOutSourceConfig: ({ config }) => {
    // ...
    set({ ...get(), ...config }, false, `受控更新:  组件配置`);
    // ...
  },
});

const createStore = create(devtools(vanillaStore, { name: "ProTableStore" }));

然后我们就可以在redux-devtools中愉快地查看数据变更了:

动图封面

可能有小伙伴会注意到,为什么我这边的状态变更还有中文名,那是因为 devtools 中间件为zustandset方法,提供了一个额外参数。只要设置好相应的set值的最后一个变量,就可以直接在devtools中看到相应的变更事件名称。

正是这样强大的分形能力,我们基于社区里做的一个 zundo 中间件,在ProEditor中提供了一个简易的撤销重做 的Demo示例。

img

而实现核心功能的代码就只有一行~

img

PS:至于一开始提到的协同能力,我在社区中也有发现中间件 zustand-middleware-yjs (不过还没尝试

❻ 多环境集成( react内外环境联动 )

实际的复杂应用中,一定会存在某些不在react环境内的状态数据,以图表、画布、3D场景最多。一旦要涉及到多环境下的状态管理,可以让人掉无数头发。

zustand说了,不慌,我已经考虑到了,useStore 上直接可以拿值,是不是很贴心~

// 官方示例

// 1. 创建Store
const useDogStore = create(() => ({ paw: true, snout: true, fur: true }))

// 2. react 环境外直接拿值
const paw = useDogStore.getState().paw

// 3. 提供外部事件订阅
const unsub1 = useDogStore.subscribe(console.log)

// 4. react 世界外更新值
useDogStore.setState({ paw: false })

const Component = () => {
  // 5. 在 react 环境内使用
  const paw = useDogStore((state) => state.paw)
  ...

虽然这个场景我还没遇到,但是一想到zustand在这种场景下也能支持,真的是让人十分心安。

img

其实还有其他不太值得单独提的点,比如zustand在测试上也相对比较容易做,直接用test-library/react-hooks即可。类型定义方面做的非常齐全……但到现在洋洋洒洒已经写了6k多字了,就不再展开了。

总结:zustand是当下复杂状态管理的最佳选择

大概从去年12月份开始,我就一直在提炼符合我理想的状态管理库的需求,到看到zustand让我眼前一亮。而通过在 pro-editor 中大半年的实践验证,我很笃定地认为,zustand就是我当下状态管理的最佳选择,甚至是大部分复杂应用的状态管理的最佳选择

本来最后还想讲讲,我是怎么样基于Zustand来做渐进式的状态管理的(从小应用到复杂应用的渐进式生长方案。然后还想拿ProEditor为例讲讲ProEditor具体的状态管理是如何逐步生长的,包括如何组织的受控模式、如何集成RxJS处理复杂交互等等,算是几个比较有意思的点。不过限于篇幅原因,这些内容估计就得留到下次了。

银弹存在吗?

在上篇《为什么是Zustand中,我总结了在ProEditor这个重交互操作的场景下对状态管理的诉求,本篇将会从具体使用的角度来详细介绍下我是怎么用zustand这一个状态管理库,解决目前我所遇到的所有状态管理诉求的。

首先想从云谦老师《数据流2022最后的总结开始说起。

所以怎么选? 没有银弹!如果站在社区开发者的角度来看。先看远程状态库是否满足需求,满足的话就无需传统数据流方案了;然后如果是非常非常简单的场景用useState + Context,复杂点的就不建议了,因为需要自行处理渲染优化,手动把Context拆很细或者尝试用use-context-selector;再看心智模型,按内部store或外部store区分选择,个人已经习惯后者,前者还在观望;如果选外部store,无兼容性要求的场景优先用类Valtio基于proxy的方案,写入数据时更符合直觉,同时拥有全自动的渲染优化,有兼容性要求时则选Zustand

其实我对上述的这个决策方案并不是非常太认同,原因是状态管理会有一个重要但容易被忽略的核心需求:在遇到更加复杂的场景时,我们能不能用当前的模式轻松地承接住? 例如:

  • 当前只是5个状态,但业务突然加了一坨复杂的业务需求,然后要膨胀到15个状态。这个时候状态还容不容易维护?性能还能保证和之前一样吗?
  • 当前是React环境,但突然业务说要加个canvas图表的需求,图表又有一些配置切换功能,还是用React ,这个时候还能愉快地共享状态么?
  • 当前是个业务应用,突然某个时候业务说要把这个应用板块抽成组件供外部复用,当前的模式能不能轻松实现受控的模式改造成组件(设想一下将语雀编辑器从应用抽取变成组件)

如果承接不住,那么就意味着推翻重写,这在我看来是不可接受的。理想的架构选型,就应该为可以预见的未来避开大部分坑,而不是遇到了再换一枪打一发。所以我自己做状态管理库决策选型的两个核心原则是: 1.这个库本身的DX好不好; 2.这个库在未来一旦要遇到复杂场景的时候,能不能用简单、低成本的方式兜住我的需求?

虽然我自己的实践有限,但我还想说zustand是银弹。

基于Zustand的渐进式状态管理

最近在给ProEditorProLayout的可视化装配器。其中一个很重要的编辑能力就是图标的选择。而这个组件也存在一点点复杂度,刚好拿来作为Zustand用法的案例,可谓是**「真·实战案例」**。 首先简单介绍一下图标选择器这个组件。它的核心用途就是让用户可以快速选择所需的图标。用户可以选择内置的Ant Design的图标,也可以使用Iconfont的图标。简单的演示如下:

动图封面

基础的展示、选择、移除、搜索

动图封面

iconfont的添加、切换、删除

为了满足上述的目标,这个组件具有下述功能:

  1. 展示图标列表;
  2. 选择、删除图标;
  3. 搜索图标;
  4. 切换antd图标和iconfont图标类目;
  5. 添加与删除Iconfont脚本(暂不准备加编辑
  6. 切换iconfont脚本展示不同的iconfont脚本下的图标;

同时,由于这个组件需要被多个场景复用,因此它需要支持非受控模式与受控模式。同时为了提升研发效率,我也希望能用devtools检查相应的状态情况。

讲完基础的需求后,一起来看看这个组件是如何通过Zustand完整实现的。

Step 1: store初始化 :State

首先拿最简单的tabs切换做一个组件tabs切换的功能。新建一个 store.ts 文件,然后写下如下代码:

import create from 'zustand';

// 注意一个小细节,建议直接将该变量直接称为 useStore
export const useStore = create(() => ({
  panelTabKey: 'antd',
})

在相应的组件(PickerPanel)中引入 useStore ,用hooks的方式即可解构获得 panelTabKey。而需要修改状态时,可直接使用 useStore.setState 即可对 panelTabKey 进行修改。这样, zustand最简单的状态管理方法就完成了~

import { Segmented } from "antd";

import { useStore } from "../store";

const PickerPanel = () => {
  const { panelTabKey } = useStore();

  return (
    // ...
    <Segmented
      value={panelTabKey}
      onChange={(key) => {
        useStore.setState({ panelTabKey: key });
      }}
      // 其他配置
      size={"small"}
      options={[
        { label: "Ant Design", value: "antd" },
        { label: "Iconfont", value: "iconfont" },
      ]}
      block
    />

    // ...
  );
};

:::info为了续统一心智,我在这里先将create中声明的状态部分,都称为State:::由于zustand默认全局单例,因此只要声明一个useStore即可在所有地方使用,不用在外层套一个Context ,非常舒心。同时 useStore 又包含了一个 setState 的方法,因此在需要React中修改状态时,可以直接使用 setState 进行状态修改。 这是zustand的最最简单的使用方式,在场景初始化的时候,这样就能直接上手使用,非常简单,直接干掉useStore + Context妥妥的。

Step 2:状态变更方法:Action

Step 1中,我们用 setState 来管理非常简单的状态,这些状态基本上用不着为其单独设定相应的变更功能。但是随着业务场景的复杂性增多,我们不可避免地会遇到存在一次操作需要变更多个状态的场景。 :::info而这些具有特定功能的状态变更方法,我统一称之为Action:::在图标选择器中,Action其中之一的体现就是选择图标的操作。选择图标这个操作,除了设定当前选中的图标以外,还需要关闭popover、清除筛选关键词(否则下次打开还是有筛选词的

动图封面

因此我们首先在store中添加三个状态:

import create from 'zustand';


export const useStore = create(() => ({
  panelTabKey: 'antd',
  iconList: ...,

  open: false,
  filterKeywords: '',
  icon: null,
})

如果我们直接用Step1的方式,大致的写法如下:

import { useStore } from '../store';

const IconList = () => {
  const { iconList } = useStore();
  return (
    <div>
      {iconList.map((icon) => (
        <IconThumbnail onClick={(icon) => {
            useStore.setState({ icon, open: false, filterKeywords: undefined });
        })} />
      ))}
    </div>
  );
};

但此时会遇到新的问题,如果我在另外一个地方也需要使用这样一段操作逻辑时,我要写两次么?当然不,这既不利于开发,也不利于维护。 所以,在这里我们需要抽取一个 selectIcon 方法专门用于选择图标这个操作,相关的状态只要都写在那里即可。而这就引出了状态管理的第二步:自定义Action。 在 store.ts 中直接声明并定义 selectIcon 函数,然后第一个入参改为set,就可以在store.ts的方法内部直接修改状态了,代码如下所示:

import create from 'zustand';


// 添加第一个入参 set
export const useStore = create((set) => ({
  panelTabKey: 'antd',
  iconList: ...,

  open: false,
  filterKeywords: '',
  icon: null,

  // 新增选择图标的 action
  selectIcon: (icon) => {
    set({ icon, open: false, filterKeywords: undefined });
    },
})

对应在 IconList 中,只需引入 selectIcon 方法即可。

import { useStore } from "../store";

const IconList = () => {
  const { iconList, selectIcon } = useStore();
  return (
    <div>
      {iconList.map((icon) => (
        <IconThumbnail onClick={selectIcon} />
      ))}
    </div>
  );
};

另外值得一提的两个小点:

  • Action支持 async/await,直接给函数方法添加async符号即可;
  • zustand默认做了变量的优化,只要是从 useStore解构获得的函数,默认是引用不变的,也就是使用zustand store的函数本身并不会造成不必要的重复渲染。

Step 3:复杂状态派生:Selector

Step2中大家应该有看到iconList这个状态,在上例中由于iconList并不是重点,因此简化了写法。但事实上在图标选择器组件中, iconList并不是一个简简单单的状态,而是一个复合的派生状态。 在选择器组件中, iconList首先需要基于 是Ant Designtab或者Iconfonttabs做原始图标数据源的进行切换,同时还需要支持相应的检索能力。而由于Ant Design TabIconfont下的list具有不同的数据结构,因此筛选逻辑的实现也是不同的。

动图封面

那在zustandStore中,这个iconList是怎么实现的呢?在这里就要介绍zustand的又一个利器: Selector 。 此selectorreduxselector的理念基本上是一致的,因此如果之前了解过zustandselectorzustand的也一样很容易理解。但从使用上来说,我认为zustandselector更加灵活易用。 首先是定义selectorselector的入参是完整的store (包含stateaction ,出参是目标对象。

import create from 'zustand';


export const useStore = create(() => ({
  // 数据源
  panelTabKey: 'antd',
  antdIconList,
  iconfontIconList,

  //...
})

// 展示用户会看到 icon list
export const displayListSelector = (s: typeof useStore) => {
  // 首先判断下来自哪个数据源
  const list = s.panelTabKey === 'iconfont' ? s.iconfontIconList : s.antdIconList
  // 解构拿到 store 中的关键词
  const { filterKeywords } = s;

  // 然后做一轮筛选判断
  return list.filter((i) => {
    if (!filterKeywords) return true;

    // 根据不同的图标类型使用不同的筛选逻辑
    switch (i.type) {
      case 'antd':
      case 'internal':
        return i.componentName.toLowerCase().includes(filterKeywords.toLowerCase());

      case 'iconfont':
        return i.props.type.toLowerCase().includes(filterKeywords.toString());
    }
  });
};

当定义完成selector后,在组件层面作为useStore的第一个入参即可:

import { useStore, displayListSelector } from "../store";

const IconList = () => {
  const { selectIcon } = useStore();
  const iconList = useStore(displayListSelector);

  return (
    <div>
      {iconList.map((icon) => (
        <IconThumbnail onClick={selectIcon} />
      ))}
    </div>
  );
};

如此一来,就完成了复杂状态的派生实现。因为useStore可以像多个hooks一样进行引入,因此我们就可以利用selector选出自己需要的各种状态,也可以多个selector间进行组合,复用通用逻辑。

多组 selector

Selector 间的组合

另外,如果用selector选择出来的变量也属于react世界中的状态,因此为了避免不必要的重复渲染,可以对复杂的对象或者数组使用isEqual方法做比较,保证它的不变性。

import { useStore, displayListSelector } from "../store";
import { isEqual } from "lodash";

const IconList = () => {
  const { selectIcon } = useStore();
  // 通过加入 isEqual 方法即可实现对 iconList 的性能优化
  const iconList = useStore(displayListSelector, isEqual);

  return (
    <div>
      {iconList.map((icon) => (
        <IconThumbnail onClick={selectIcon} />
      ))}
    </div>
  );
};

最后,由于selector本身的定义只是个纯函数,也能非常方便地集成单元测试。

Step 4:结构组织与类型定义

经过一部分功能开发,一开始简单的 store.ts 文件开始变得很长了,同时估计也开始遇到类型定义不准确或找不到的情况了。那这对于后续项目的规模化发展非常不利,是时候做一次组织与整理了。

import create from 'zustand';


// 添加第一个入参 set
export const useStore = create((set) => ({
  panelTabKey: 'antd',
  iconList: ...,
  antdIconList,

  open: false,
  filterKeywords: '',
  icon: null,


  iconfontScripts: [],
  iconfontIconList: [],
  onIconChange: null,

  // 新增选择图标方法
  selectIcon: (icon) => {
    set({ icon, open: false, filterKeywords: undefined });
    },
  // 移除图标方法
  resetIcon: () => {
    set({ icon: undefined });
  },

  addSript:()=>{ /*...*/ },
  updateScripts:()=>{ /*...*/ },
  removeScripts:()=>{ /*...*/ },
  selectScript:async (url)=>{ /*...*/ }
  // ...
})

// 展示用户会看到 icon list
export const displayListSelector = (s: typeof useStore) => {
  // 首先判断下来自哪个数据源
  const list = s.panelTabKey === 'iconfont' ? s.iconfontIconList : s.antdIconList
  // 解构拿到 store 中的关键词
  const { filterKeywords } = s;

  // 然后做一轮筛选判断
  return list.filter((i) => {
    if (!filterKeywords) return true;

    // 根据不同的图标类型使用不同的筛选逻辑
    switch (i.type) {
      case 'antd':
      case 'internal':
        return i.componentName.toLowerCase().includes(filterKeywords.toLowerCase());

      case 'iconfont':
        return i.props.type.toLowerCase().includes(filterKeywords.toString());
    }
  });
};

所以在我建议在Step4开始,就要对ZustandStore进行更加合理地划分。 首先是从 store.ts 重构为 store 文件夹,目录结构如下:

./store
├── createStore.ts        // Action 与 store
├── selectors.ts          // 状态派生
├── initialState.ts       // State 类型定义与 初始状态
└── index.ts

如此划分的依据本质上还是基于StateActionSelector的三者切分:

  • initialState.ts:负责State —— 添加状态类型与初始化状态值;
  • createStore.ts: 负责书写创建Store的方法与Action方法;
  • selectors.ts: 负责Selector ——派生类选择器逻辑;

首先来看看 initialState ,这个文件中主要用于定于并导出后续在Store所有需要的状态。导出的部分包含两个: State 类型定义与 初始状态 initialState。将StateinitialState定义在一个文件中会有一个好处:类型跳转会直接指向到这里,方便添加类型与类型的初始值。 由于state单独新建了一个文件,因此哪怕后续状态再多,也能在这一个文件中看得清清楚楚。

import type {
  ExternalScripts,
  IconfontIcon,
  IconUnit,
  ReactIcon,
} from "../types";
import { antdIconList } from "../contents/antdIcons";

export interface State {
  iconfontScripts: ExternalScripts[];
  icon?: IconUnit;
  showEditor: boolean;

  open: boolean;
  panelTabKey: "antd" | "iconfont";
  filterKeywords?: string;

  activeIconfontScript?: string;
  antdIconList: ReactIcon[];
  iconfontIconList: IconfontIcon[];
}

export const initialState: State = {
  open: false,
  showEditor: false,
  panelTabKey: "antd",
  filterKeywords: "",
  antdIconList,

  iconfontScripts: [],
  iconfontIconList: [],
  onIconChange: null,
};

再来看看 createStore ,这个文件由于包含了ActionStore,会稍显复杂一点,但是核心逻辑还是比较简单的。

import create from "zustand";

import type { State } from "./initialState";
import { initialState } from "./initialState";

interface Action {
  resetIcon: () => void;
  togglePanel: () => void;
  selectIcon: (icon: IconUnit) => void;
  removeScripts: (url: string) => void;
  selectScript: (url: string) => void;
  toggleEditor: (visible: boolean) => void;
  addScript: (script: ExternalScripts) => void;
  updateScripts: (scripts: ExternalScripts[]) => void;
}

export type Store = State & Action;

export const useStore = create<Store>((set, get) => ({
  ...initialState,

  resetIcon: () => {
    set({ icon: undefined });
  },

  selectIcon: (icon) => {
    set({ icon, open: false, filterKeywords: undefined });
  },

  addSript: () => {
    /*...*/
  },
  updateScripts: () => {
    /*...*/
  },
  removeScripts: () => {
    /*...*/
  },
  selectScript: async (url) => {
    /*...*/
  },
}));

它做了这么几件事:

  1. 定义了storeAction的类型,然后将StateAction合并为Store类型,并导出了Store的类型(比较重要
  2. create方法添加了Store的类型,让store内部识别到自己这个store包含了哪些方法;
  3. initialState 解构导入store(原来定义state的部分已经抽出去到initialState里了
  4. useStore里逐一实现Action中定义的方法;

所以将从 store.ts 重构到 createStore.ts ,基本上只有补充类型定义的工作量。

接下来再看下selectors,这个文件很简单,只需要导入Store的类型,然后逐一导出相应的selector即可。

import type { Store } from "./createStore";
import type { IconUnit } from "../types";

export const isEmptyIconfontScripts = (s: Store) =>
  s.iconfontScripts.length === 0;

export const selectedListSelector = (s: Store): IconUnit[] =>
  s.panelTabKey === "iconfont" ? s.iconfontIconList : s.antdIconList;

export const isEmptyIconListSelector = (s: Store) =>
  selectedListSelector(s).length === 0;

export const displayListSelector = (s: Store) => {
  const list = selectedListSelector(s);
  const { filterKeywords } = s;

  return list.filter((i) => {
    if (!filterKeywords) return true;

    switch (i.type) {
      case "antd":
      case "internal":
        return i.componentName
          .toLowerCase()
          .includes(filterKeywords.toLowerCase());

      case "iconfont":
        return i.props.type.toLowerCase().includes(filterKeywords.toString());
    }
  });
};

最后在 index.ts 中输出相应的方法和类型即可:

export { useStore } from "./createStore";
export type { Store } from "./createStore";
export type { State } from "./initialState";
export * from "./selectors";

如此一来,我们通过 将store.ts单一职责的文件,拆分成各司其职的多个文件后,就初步解决了接下来可能的状态大量扩展的问题与类型定义不准确的问题,基本上可以保证项目的可维护性。

Step 5:复杂Action交互:get()

Step1~Step3可能在很大程度上就能满足大部分场景的状态管理诉求,从Step4开始,其实意味着状态管理的复杂性开始上升,因此Step4也是为了驾驭复杂性而做的一个铺垫。 在图标选择器这个组件中,ant design的图标选择部分并没有太复杂的逻辑,最复杂的部分也只是关键词搜索的展示。但iconfont部分的图标不同,这一部分由于存在用户的输入和多个数据源的配置、切换,复杂性是急剧上升的。 譬如存在的展示状态就有:空数据 、空数据但要添加数据源、有数据未选中态、有数据选中态等总共7种状态。所以在状态管理的复杂性也是急剧上升。

img

zustand如何做到有效地收敛这些操作呢?核心思路就是基于用户行为分层拆解一级承接的Action ,最后在多个方法的基础上统一抽取原子操作Action

img

首先是要识别用户操作维度上的行为,在Iconfont这个场景下,用户的行为有三类:添加数据源选择数据源移除数据源

  • 先来看「添加数据源」 :用户感知到的添加数据源这个行为看起来简单,但在数据流上其实包含四个步骤:❶ 显示表单-> ❷ 将数据源添加到数据源数组中-> ❸ 隐藏表单-> ❹ 选中添加的数据源。

动图封面

  • 再来看「选择数据源:选择数据源就是当用户存在多个Iconfont图标源的时候,用户可以按自己的诉求切换不同的Iconfont数据源;

动图封面

  • 最后再看下「移除数据源:移除数据源看似很简单,但是其实也存在坑。即:移除当前选中的数据源时,怎么处理选中态的逻辑?基于良好的用户体验考虑,我们会自动帮用户切换一个选中的数据源。但这里就会有边界问题:如果删除的数据源第一个,那么应该往后选择,如果删除的是最后一个数据源,那么应该往前选择。 |

动图封面

删除第一个

|

动图封面

删除最后一个

那这样的功能从数据流来看,移除数据源会包含三个阶段:❶ 从数据源数组中移除相应项-> ❷ 决策需要选中哪个数据源-> ❸ 选中相应的数据源。

基于上述的分析,我们可以发现三层方法会存在一些重复的部分,主要是对数据源数组的更新与设定激活数据源。所以相应的Action可以拆解如下:

img

所以我们在store中定义这些这些方法:

import create from "zustand";

import type { State } from "./initialState";
import { initialState } from "./initialState";

interface Action {
  /* 一级 action */
  addScript: (script: ExternalScripts) => void;
  removeScripts: (url: string) => void;
  selectScript: (url: string) => void;

  /* 原子操作 action */
  updateScripts: (scripts: ExternalScripts[]) => void;
  toggleForm: (visible: boolean) => void;
}

export type Store = State & Action;

来看下具体的实现,在zustand中能实现上述架构的核心能力在于一个 get() 方法,能从自身中拿到所有的状态(State & Action

// ...

export type Store = State & Action;

// create 函数的第二个参数 get 方法
export const useStore = create<Store>((set, get) => ({
  ...initialState,

  // 用户行为 action //

  addScript: (script) => {
    // 从 get() 中就可以拿到这个 store 的所有的状态与方法
    const { selectScript, iconfontScripts, updateScripts, toggleForm } = get();

    // 1. 隐藏 Form
    toggleForm(false);

    // 2. 更新数据源
    updateScripts(
      produce(iconfontScripts, (draft) => {
        if (!draft.find((i) => i.url === script.url)) {
          draft.push(script);
        }
      })
    );

    // 3. 选择脚本
    selectScript(script.url);
  },
  removeScripts: (url) => {
    const { iconfontScripts, selectScript, updateScripts } = get();

    const nextIconfontScripts = iconfontScripts.filter((i) => i.url !== url);

    // 找到临近的图标库并选中

    const currentIndex = iconfontScripts.findIndex((i) => i.url === url);

    const nextIndex =
      currentIndex === 0
        ? 0
        : nextIconfontScripts.length <= currentIndex + 1
        ? currentIndex - 1
        : currentIndex;

    const nextScript = nextIconfontScripts[nextIndex]?.url;

    updateScripts(nextIconfontScripts);

    selectScript(nextScript);
  },

  // 原子操作方法 //

  toggleForm: (visible) => {
    set((s) => ({
      ...s,
      showForm: typeof visible === "undefined" ? !s.showForm : visible,
    }));
  },
  selectScript: async (url) => {
    // 如果没有 url ,就说明是取消选择
    if (!url) {
      set({ activeIconfontScript: "", iconfontIconList: [] });
      return;
    }

    // 2. 一个异步方法获取脚本中的图标列表
    const iconfontList = await fetchIconList(url);

    // 3. 设定选中后的数据更新
    set({
      activeIconfontScript: url,
      iconfontIconList: iconfontList.map((i) => ({
        type: "iconfont",
        componentName: iconfontScripts.name,
        scriptUrl: url,
        props: { type: i },
      })),
    });
  },
  updateScripts: (scripts) => {
    const { iconfontScripts } = get();

    if (isEqual(iconfontScripts, scripts)) return;
    set({ iconfontScripts: scripts });
  },
}));

当完成相应的功能实现后,只需要在相应的触发入口中添加方法即可。

const IconfontScripts: FC = memo(() => {
  const {
    iconfontScripts,
    showForm,
    activeIconfontScript,
    removeScripts,
    selectScript,
    toggleEditor,
  } = useStore();
  const isEmptyScripts = useStore(isEmptyIconfontScripts);

  return (
    <Flexbox gap={8}>
      <Flexbox gap={4} horizontal>
        {showForm ? (
          <ActionIcon
            onClick={() => toggleEditor(false)}
            icon={<UpOutlined />}
          />
        ) : (
          <Tag
            onClick={() => {
              toggleEditor(true);
            }}
          >
            <PlusOutlined /> 添加
          </Tag>
        )}
        <Flexbox horizontal>
          {iconfontScripts.map((s) => {
            const checked = s.url === activeIconfontScript;

            return (
              <Tag
                onClose={() => {
                  removeScripts(s.url);
                }}
                onClick={() => {
                  selectScript(checked ? "" : s.url);
                }}
              >
                {s.name}
              </Tag>
            );
          })}
        </Flexbox>
      </Flexbox>
      {showForm ? <ScriptEditor /> : null}
    </Flexbox>
  );
});

export default IconfontScripts;

基于这样的一种模式,哪怕是这样一个复杂的组件,在实现层面的研发心智仍然非常简单:

  • React层面仍然只是一个渲染层;
  • 复杂的状态逻辑仍然以hooks式的模式进行引入;
  • 复杂的入口方法通过拆分子Action进行组合与复用;

而且我们在Step2中已经知道,在这种模式下,所有的Action都是默认不需要包useCallback~

Step 6:从应用迈向组件:ContextStoreUpdater

不知道有没有小伙伴从Step1开始就在纳闷,你不是说这是个组件吗?为啥一直没提受控模式呢?在你这个模式下,不是就变成全局单例了么?怎么搞成组件?那这一步不就来了么,就让我们来看看在zustand模式下,一个已经略显复杂的应用,如何轻轻松松变成一个受控的业务组件。 因为zustand是默认全局单例,所以如果需要变成组件,那么一定需要使用Context来隔离多个实例。而 这其中的关键,就是zustand提供的 createContext 方法。 这个改造分为四步:

img

第一步: 创建Context并添加Provider先在 createStore.ts

import create from 'zustand';
import createContext from 'zustand/context';
import type { StoreApi } from 'zustand';

import type { State } from './initialState';
import { initialState } from './initialState';

interface Action {
  // *** /
}

export type Store = State & Action;

// 将 useStore 改为 createStore, 并把它改为 create 方法
export const createStore = ()=> create<Store>((set, get) => ({
  ...initialState,

  resetIcon: () => {
    set({ icon: undefined });
  },

  selectIcon: (icon) => {
    set({ icon, open: false, filterKeywords: undefined });
  },

  addSript:()=>{ /*...*/ },
  updateScripts:()=>{ /*...*/ },
  removeScripts:()=>{ /*...*/ },
  selectScript:async (url)=>{ /*...*/ }
}));

// 新建并导出一下 Provider、useStore、useStoreApi 三个对象
export const { Provider, useStore, useStoreApi } = createContext<StoreApi<Store>>();
import type { FC } from 'react';
import React, { memo } from 'react';
import App from './App';

import { Provider, createStore } from '../store';

type IconPickerProps = StoreUpdaterProps;

const IconPicker: FC<IconPickerProps> = (props) => {
  return (
    <Provider createStore={createStore}>
      <App /> /* <- 这个App就是之前的引用入口 */
    </Provider>
  );
};
export default memo(IconPicker);

第二步:创建并添加受控更新组件 **StoreUpdater**

首先在组件入口处添加 StoreUpdater 组件。

import type { FC } from "react";
import React, { memo } from "react";
import App from "./App";
import StoreUpdater from "./StoreUpdater";
import type { StoreUpdaterProps } from "./StoreUpdater";

import { Provider, createStore } from "../store";

type IconPickerProps = StoreUpdaterProps;
const IconPicker: FC<IconPickerProps> = (props) => {
  return (
    <Provider createStore={createStore}>
      <App />
      <StoreUpdater {...props} />
    </Provider>
  );
};
export default memo(IconPicker);

StoreUpdater具体是干什么的? 看下面这张图,我想大家就懂了。

img

简单来说,就是通过 StoreUpdater 这个组件,做到外部props和内部状态的隔离。这样一来,当没有外部props时,我们直接可以把这个App当成普通应用。而当有外部的props时,可以通过 StoreUpdater 实现外部状态的受控。利用这样的思想,我们就可以很简单地把一个App改造成受控的业务组件。这种架构模式,也称为「分形架构

如果子组件能够以同样的结构,作为一个应用使用,这样的结构就是分形架构。在分形架构下,每个应用都可以变成组件,被更大的应用合并消费。

具体来看看代码:

import type { FC } from "react";

import type { State } from "../store";
import type { IconUnit, ExternalScripts } from "../types";

import { useStoreApi } from "../store";

/**
 * 更新方法
 */
export const useStoreUpdater = (
  key: keyof T,
  value: any,
  deps = [value],
  updater?
) => {
  const store = useStoreApi();

  useEffect(() => {
    if (typeof value !== "undefined") {
      store.setState({ [key]: value });
    }
  }, deps);
};

export interface StoreUpdaterProps
  extends Partial<
    Pick<
      State,
      "icon" | "onIconChange" | "iconfontScripts" | "onIconfontScriptsChange"
    >
  > {
  defaultIcon?: IconUnit;
  defaultIconfontScripts?: ExternalScripts[];
  defaultActiveScripts?: ExternalScripts[];
}

const StoreUpdater: FC<StoreUpdaterProps> = ({
  icon,
  defaultIcon,
  iconfontScripts,

  defaultIconfontScripts,
  onIconChange,
  onIconfontScriptsChange,
}) => {
  useStoreUpdater("icon", defaultIcon, []);
  useStoreUpdater("icon", icon);
  useStoreUpdater("onIconChange", onIconChange);

  useStoreUpdater("iconfontScripts", iconfontScripts);
  useStoreUpdater("iconfontScripts", defaultIconfontScripts, []);
  useStoreUpdater("onIconfontScriptsChange", onIconfontScriptsChange);

  return null;
};
export default StoreUpdater;

StoreUpdater 这个组件中,核心分为三个部分:

  • useStoreUpdater :将外部的props同步到store内部的方法;
  • StoreUpdaterProps:从storeStatepick出需要受控的状态,并相应补充defaultXXprops
  • StoreUpdater:逐一补充调用外部组件props的受控状态,将外部props更新到store的内部状态中;

针对受控组件来说,要实现一个状态props的受控,一定会有的孪生的两个props。例如一个propsvalue ,一定会有一个defaultValue和一个onValueChange,才能满足所有和这个value相关的场景诉求。因此我在 StoreUpdater 中是也完全基于这个规则书写受控代码。不过这里有个很有意思的点:就是在受控模式下,把onChange也当成store的自持的状态去思考。所有的onChange类方法,只有两种状态 nullfunction。 这样就能在store内部很轻松地完成受控方法的集成。

zustand在这个过程中发挥最关键一点的hooks叫做 useStoreApi。这个props可能大家在Step1~ Step5中都没看到过,因为这个hooks只在context的场景下出现。它的功能就是获得相应 Context 下的 useStore 方法。如果直接使用 useStore,那么是获得不到挂在在useStore上的变量的。 大家是否还记得Step1中的 useStore.setState({ ... })这个方法?它在这个场景下发挥了巨大的作用,大大减少了更新受控props的代码量。这是我们需要在这里使用 useStoreApi 的原因。

这一步写完之后,组件接受外部props ,并受控的部分就完成了。最后就只剩内部状态变更需要onChange出来了。

第三步:在相应的Action里添加onChange方法 在第二步中看到,我们需要在StoreState中把onChange方法作为状态自持,因此在initalState文件中,就需要补充相应的类型定义和初始值:

import type {
  ExternalScripts,
  IconfontIcon,
  IconUnit,
  ReactIcon,
} from "../types";
import { antdIconList } from "../contents/antdIcons";

export interface State {
  iconfontScripts: ExternalScripts[];
  icon?: IconUnit;
  showForm: boolean;
  /**
   * 开启面板
   */
  open: boolean;
  panelTabKey: "antd" | "iconfont";

  filterKeywords?: string;

  activeIconfontScript?: string;
  antdIconList: ReactIcon[];
  iconfontIconList: IconfontIcon[];

  // 外部状态
  onIconChange?: (icon: IconUnit) => void;
  onIconfontScriptsChange?: (iconfontScripts: ExternalScripts[]) => void;
}

export const initialState: State = {
  open: false,
  showForm: false,
  panelTabKey: "antd",
  filterKeywords: "",
  antdIconList,

  iconfontScripts: [],
  iconfontIconList: [],

  onIconChange: null,
  onIconfontScriptsChange: null,
};

而因为我们在Step5中通过收敛了一些原子级的Action,基本做到了一个State有一个对应的Action,因此只需要相应的Action处添加受控更新的onChange方法即可。

// ...

export type Store = State & Action;

export const createStore = () =>
  create<Store>((set, get) => ({
    ...initialState,

    selectIcon: (icon) => {
      set({ icon, open: false, filterKeywords: undefined });

      // 受控更新 icon
      get().onIconChange?.(icon);
    },
    // 用户行为 action //

    addScript: (script) => {
      // 从 get() 中就可以拿到这个 store 的所有的状态与方法
      const { selectScript, iconfontScripts, updateScripts, toggleForm } =
        get();

      // 1. 隐藏 Form
      toggleForm(false);

      // 2. 更新数据源
      updateScripts(
        produce(iconfontScripts, (draft) => {
          if (!draft.find((i) => i.url === script.url)) {
            draft.push(script);
          }
        })
      );

      // 3. 选择脚本
      selectScript(script.url);
    },
    removeScripts: (url) => {
      const { iconfontScripts, selectScript, updateScripts } = get();

      const nextIconfontScripts = iconfontScripts.filter((i) => i.url !== url);

      // 找到临近的图标库并选中

      const currentIndex = iconfontScripts.findIndex((i) => i.url === url);

      const nextIndex =
        currentIndex === 0
          ? 0
          : nextIconfontScripts.length <= currentIndex + 1
          ? currentIndex - 1
          : currentIndex;

      const nextScript = nextIconfontScripts[nextIndex]?.url;

      updateScripts(nextIconfontScripts);

      selectScript(nextScript);
    },

    // 原子操作方法 //

    toggleForm: (visible) => {
      set((s) => ({
        ...s,
        showForm: typeof visible === "undefined" ? !s.showForm : visible,
      }));
    },
    selectScript: async (url) => {
      // 如果没有 url ,就说明是取消选择
      if (!url) {
        set({ activeIconfontScript: "", iconfontIconList: [] });
        return;
      }

      // 2. 一个异步方法获取脚本中的图标列表
      const iconfontList = await fetchIconList(url);

      // 3. 设定选中后的数据更新
      set({
        activeIconfontScript: url,
        iconfontIconList: iconfontList.map((i) => ({
          type: "iconfont",
          componentName: iconfontScripts.name,
          scriptUrl: url,
          props: { type: i },
        })),
      });
    },
    updateScripts: (scripts) => {
      const { iconfontScripts } = get();

      if (isEqual(iconfontScripts, scripts)) return;
      set({ iconfontScripts: scripts });

      // 受控更新 IconfontScripts
      get().onIconfontScriptsChange?.(scripts);
    },
  }));

如此一来,组件的受控就完成了。

(可选)第四步:查找useStore.setState用法,补充useStoreApi如果有一些状态非常简单,从写下的一开始就始终是 useStore.setState 的写法,那么这些写法在组件化之后需要做一点点小调整。因为useStore是完全来自于context下的useStore,因此会丢失setState的相关方法。因此需要额外引入 useStoreApi ,并用storeApi来实施setState。这可能算是算zustand从应用迁移到组件的一点点小瑕疵。

import { useStore, useStoreApi } from '../store';

const IconList = () => {
  const { iconList } = useStore();

  const storeApi = useStoreApi()

  return (
    <div>
      {iconList.map((icon) => (
        <IconThumbnail onClick={(icon) => {
            storeApi.setState({ icon, open: false, filterKeywords: undefined });
        })} />
      ))}
    </div>
  );
};

不过如果是真正的复杂应用,经历过Step1~Step5之后,估计大部分状态变更都会收敛到Store中,因此如果需要修改setState的部分,在我实际使用下来并不算太多。 最后来看下这样的一个效果:

动图封面

可以看到,基于zustand的这样的开发模式,一个业务应用可以非常简单地迁移成为一个受控的业务组件,且任何一个需要对外暴露的数据源,都可以非常轻松地做到受控。而这是我认为zustand作为状态管理库极其好用的一点。

PS:这个 StoreUpdater 的用法我也是翻了react-flow才了解到的。它的 StoreUpdater 比我这个组件可是多多了。而这样的props,使得react-flow这个复杂组件的灵活度和可玩性得到了保障(有兴趣的同学可以看看它的源码 传送门->

Step 7:性能优化:又是selector ?

作为一个制作精良的组件,性能上一定不能拉胯。因此我们需要来做下优化了。最近云谦老师写了篇《关于React Re-Render,可以看到如果要裸写hooks做性能优化,得学习一堆基础知识和踩一堆坑,最后代码里一坨一坨的useXXX,心智负担非常重。那在zustand中怎么做性能优化呢?

首先,在Step2中我们已经知道,所有zustandaction都默认不会造成重复渲染,因此,理论上只有state会造成重复渲染。我们来看下实际情况。 首先我使用 useWhyDidYouUpdatehooks来检查并确认PickerPanel组件的stateaction是否会会发变化,从下图中可以看到,无论是resetIcon还是还是storeApi ,它们的引用在其他状态变化下,都保持不变,没有造成重复渲染。

动图封面

但在上图中我们可以看到一个挺诡异的现象。就是明明 useWhyDidYouUpdate 已经包含了所有的该组件使用的状态,在修改搜索关键词时,State都没有变化,但是Segment那部分却可以看到有重复渲染。那这是为什么? 我们来改一下,写法,将store从单独定义为一个变量,然后用 useWhyDidYouUpdate 来检查变更,可以看到下面这样的情况:

动图封面

即当关键词state修改了,就会造成store的变化。而store的变化就会触发当前这个界面的重新渲染。哎?那这个不就是和context一模一样了么?对,你想的没错, zustand并不像valtio这样会自动收集依赖,并做性能优化。 所以我们是需要手动搞一轮优化的。那咋搞呢?思路上其实也非常简单:既然我在PickerPanel这个组件中只关心{ panelTabKey, icon, resetIcon } 这几个状态,那我「手动」做一个依赖收集不就好了么?那怎么手动做呢? 还记得step2中的selector吗?又轮到它出场了。

动图封面

只需要利用zustanduseStoreselector能力,配合zustand默认提供的 shallow 浅比较能力。我们就能实现「人工的依赖收集。如此一来,性能优化也就做好了。 我们来看看优化前和优化后的代码区别:

PickerPanel优化前:

import { useStore, useStoreApi } from '../store';


const PickerPanel = () => {
  const { panelTabKey, icon, resetIcon } = useStore();

  // 其他

  return <>{ /*... */ }<>
}

PickerPanel优化后:

import shallow from 'zustand/shallow';

import type { Store } from '../store';
import { useStore, useStoreApi } from '../store';

const selector = (s: Store) => ({
  panelTabKey: s.panelTabKey,
  icon: s.icon,
  resetIcon: s.resetIcon,
});

const PickerPanel = () => {
  const { panelTabKey, icon, resetIcon } = useStore(selector,shallow);

  // 其他

  return <>{ /*... */ }<>
}

可以看到,除了多一个几乎一样的selector和一个shallow,其他代码没有任何区别,但是性能优化就是这么做好了。 那这是基于zustand selector的写法可以做到的渐进式性能优化「需要优化?加个selector就好~」这样的研发心智,可以让业务开发有很多选择,譬如:

  • 前期撒开来默认解构useStore,不必担心未来的性能优化难题。等发现某些地方真的需要优化时,相应的套上selector就好;
  • 反正只是加个selector,也可以写完应用定型后也可以顺手加一下;
  • 既然selector可以做优化,那我干脆全部都直接 const x = useStore(s=>s.x),这样引入好,也直接优化完了。

Step 8:研发增强:Devtools

Store复杂度到现在这样之后,接下来每一步debug都有可能变得比较麻烦,因此我们可以集成一下devtools,将Store研发模式变得更加可视化,做到可控。 而写法也非常简单,只需在create方法下包一个devtools即可,并在create后多一个()执行。

// ...
import { devtools } from "zustand/middleware";

export type Store = State & Action;

// 多一个函数执行,然后包裹 devtools
export const createStore = () =>
  create<Store>()(
    devtools(
      (set, get) => ({
        ...initialState,

        // ... action
      }),
      { name: "IconPicker" }
    )
  );

如此一来,我们就能够使用redux-dev-tools可视化地查看IconPicker的数据流了。

img

image.png

不过大家可能会发现,这个时候每一次的数据变更,都是是anoymous的变更说明,那有没有可能让每条变更都更加语义化呢?可以! 只需在set方法的第三个参数中添加更新说明文本,就可以让devtools识别到这项状态变更。

// ...
import "";

export type Store = State & Action;

// 多一个函数执行,然后包裹 devtools
export const createStore = () =>
  create<Store>()(
    devtools(
      (set, get) => ({
        ...initialState,

        // ... action
        selectIcon: (icon) => {
          set(
            { icon, open: false, filterKeywords: undefined },
            false,
            "选择 Icon"
          );

          get().onIconChange?.(icon);
        },
      }),
      { name: "IconPicker" }
    )
  );

img

基于这样的写法,我们甚至可以畅享一个面向用户的历史记录能力~

还有吗?

写完上面Step1~Step 8,基本上绝大多数状态管理的需求就都能满足了。但在一些各种边界条件与复杂场景下,一定还是会有各种奇奇怪怪的诉求的。所以我在这里再列了一些自己zustand的其他用法,但不再细讲:

  • 集成redux reducerreact useReducer的写法;
  • 觉得事件处理麻烦,需要借用rxjs简化事件流处理;
  • 需要结合一些请求库,比如swr的使用方式,将hooks集成到store中;
  • 结合persist做本地数据缓存的方式;
  • 结合社区库zundo简单实现的一个历史记录功能;
  • 利用subscribe监听状态变更,自动更新内部状态;
  • 单一store的切片化;
  • 集成一些复杂三方库(例如y-js

写到这里,就基本上把这大半年所有基于zustand的状态管理经验写完了。希望对大家做状态管理的相关决策有些帮助吧~

上一页