精读《react-easy-state 源码》

1. 引言

react-easy-state 是个比较有趣的库,利用 Proxy 建立了一个很是易用的全局数据流管理方式。前端

import React from "react";
import { store, view } from "react-easy-state";

const counter = store({ num: 0 });
const increment = () => counter.num++;

export default view(() => <button onClick={increment}>{counter.num}</button>);
复制代码

上手很是轻松,经过 store 建立一个数据对象,这个对象被任何 React 组件使用时,都会自动创建双向绑定,任何对这个对象的修改,都会让使用了这个对象的组件重渲染。react

固然,为了实现这一点,须要对全部组件包裹一层 viewgit

2. 精读

这个库利用了 nx-js/observer-util 作 Reaction 基础 API,其余核心功能分别是 store view batch,因此咱们就从这四个点进行解读。github

Reaction

这个单词名叫 “反应”,是实现双向绑定库的最基本功能单元。微信

拥有最基本的两个单词和一个概念:observable observe 与自动触发执行的特性。app

import { observable, observe } from "@nx-js/observer-util";

const counter = observable({ num: 0 });
const countLogger = observe(() => console.log(counter.num));

// 会自动触发 countLogger 函数内回调函数的执行。
counter.num++;
复制代码

在第 35 期精读 精读《dob - 框架实现》 “抽丝剥茧,实现依赖追踪” 一节中有详细介绍实现原理,这里就不赘述了。框架

有了一个具备反应特性的函数,与一个能够 “触发反应” 的对象,那么实现双向绑定更新 View 就不远了。函数

store

react-easy-state 的 store 就是 observable(obj) 包装一下,惟一不一样是,因为支持本地数据:工具

import React from 'react'
import { view, store } from 'react-easy-state'

export default view(() => {
  const counter = store({ num: 0 })
  const increment = () => counter.num++
  return <button={increment}>{counter.num}</div>
})
复制代码

因此当监测到在 React 组件内部建立 store 且是 Hooks 环境时,会返回:ui

return useMemo(() => observable(obj), []);
复制代码

这是由于 React Hooks 场景下的 Function Component 每次渲染都会从新建立 Store,会致使死循环。所以利用 useMemo 并将依赖置为 [] 使代码在全部渲染周期内,只在初始化执行一次。

更多 Hooks 深刻解读,能够阅读 精读《useEffect 彻底指南》

view

根据 Function Component 与 Class Component 的不一样,分别进行两种处理,本文主要介绍对 Function Component 的处理方式,由于笔者推荐使用 Function Component 风格。

首先最外层会套上 memo,这相似 PureComponent 的效果:

return memo(/**/);
复制代码

而后构造一个 forceUpdate 用来强制渲染组件:

const [, forceUpdate] = useState();
复制代码

以后,只要利用 observe 包裹组件便可,须要注意两点:

  1. 使用刚才建立的 forceUpdatestore 修改时调用。
  2. observe 初始化不要执行,由于初始化组件本身会渲染一次,再渲染一次就会形成浪费。

因此做者经过 scheduler lazy 两个参数完成了这两件事:

const render = useMemo(
  () =>
    observe(Comp, {
      scheduler: () => setState({}),
      lazy: true
    }),
  []
);

return render;
复制代码

最后别忘了在组件销毁时取消监听:

useEffect(() => {
  return () => unobserve(render);
}, []);
复制代码

batch

这也是双向绑定数据流必须解决的经典问题,批量更新合并。

因为修改对象就触发渲染,这个过程太自动化了,以致于咱们都没有机会告诉工具,连续的几回修改可否合并起来只触发一次渲染。 尤为是 For 循环修改变量时,若是不能合并更新,在某些场景下代码几乎是不可用的。

因此 batch 就是为解决这个问题诞生的,让咱们有机会控制合并更新的时机:

import React from "react";
import { view, store, batch } from "react-easy-state";

const user = store({ name: "Bob", age: 30 });

function mutateUser() {
  // this makes sure the state changes will cause maximum one re-render,
  // no matter where this function is getting invoked from
  batch(() => {
    user.name = "Ann";
    user.age = 32;
  });
}

export default view(() => (
  <div> name: {user.name}, age: {user.age} </div>
));
复制代码

react-easy-state 经过 scheduler 模块完成 batch 功能,核心代码只有五行:

export function batch(fn, ctx, args) {
  let result;
  unstable_batchedUpdates(() => (result = fn.apply(ctx, args)));
  return result;
}
复制代码

利用 unstable_batchedUpdates,能够保证在其内执行的函数都不会触发更新,也就是以前建立的 forceUpdate 虽然被调用,可是失效了,等回调执行完毕时再一块儿批量更新。

同时代码里还对 setTimeout setInterval addEventListener WebSocket 等公共方法进行了 batch 包装,让这些回调函数中自带 batch 效果。

4. 总结

好了,react-easy-state 神奇的效果解释完了,但愿你们在使用第三方库的时候都能理解背后的原理。

PS:最后,笔者目前不推荐在 Function Component 模式下使用任何三方数据流库,由于官方功能已经足够好用了!

讨论地址是:精读《react-easy-state》 · Issue #144 · dt-fe/weekly

若是你想参与讨论,请 点击这里,每周都有新的主题,周末或周一发布。前端精读 - 帮你筛选靠谱的内容。

关注 前端精读微信公众号

special Sponsors

版权声明:自由转载-非商用-非衍生-保持署名(创意共享 3.0 许可证

相关文章
相关标签/搜索