站长家园(原代码之家)(www.adminjie.com)网站源码,微信源码,游戏源码,商业源码分享平台。
当前位置:网站首页 技术文章 网络编程 正文

大佬封装React Context Composer的详细步骤(分享)

时间:2021-12-23 [网络编程]作者:fabuyuan 浏览:7 次

本文由composer教程栏目给大家介绍大佬是如何一步步封装一个React Context Composer,希望对需要的朋友有所帮助!

我是如何一步步封装一个React Context Composer?

动机

React的状态管理方案有很多,比如Redux、Mobx、Recoil等,目前我只体验过Redux,觉得还是比较笨重一点。因为平时写Hooks比较多,所以我比较倾向于使用Context Provider配合useContext这个hook来做,这样也易于状态的拆分与组合。这里,我们不讨论各家状态管理方案的优劣,将目光聚焦于在使用Context时遇到的一个多层嵌套的问题。

下图,是我最近在写的一个taro + react hooks + ts项目抽离出来的一些代码。我对一些全局状态进行了拆分(拆分的目的是为了减少不必要的重新渲染),然后再把它们嵌套起来。这种写法让我回想起了曾经被回调地狱支配的感觉,很难受。因此,我想到了自己去封一个高阶组件,从写法上把结构“扁平化”。

<LoadingContext.Provider value={{ loading, setLoading }}>
  <UserDataContext.Provider value={{ name: "ascodelife", age: 25 }}>
    <ThemeContext.Provider value={"light"}>
    {/* ....more Providers as long as you want */}
    </ThemeContext.Provider>
  </UserDataContext.Provider>
</LoadingContext.Provider>

最易得的方案

这里,我很快的就写出了第一种方案,借助reduceRight去完成Provider的嵌套。

这里用reduceRight而不用reduce的原因是,我们更加习惯从外层到内层的书写顺序。

// ContextComposer.tsx
import React from 'react';
type IContextComposerProps = {
  contexts: { context: React.Context<any>; value: any }[];
};
const ContextComposer: React.FC<IContextComposerProps> = ({ contexts, children }) => {
  return (
    <>
      {contexts.reduceRight((child, parent) => {
        const { context, value } = parent;
        return <context.Provider value={value}>{child}</context.Provider>;
      }, children)}
    </>
  );
};
export default ContextComposer;
// App.tsx
<ContextComposer
  contexts={[
    { context: ThemeContext, value: "light" },
    { context: UserDataContext, value: { name: "ascodelife", age: 25 } },
    { context: LoadingContext, value: { loading, setLoading } },
  ]}>
    { children }
</ContextComposer>

实际体验后发现,虽然说能用是能用,但是开发体验差那么一点。它的问题在于,组件入参时传的value是any类型,这就意味着放弃了ts的静态类型检查。在传参时,由于不会对value做静态类型检查,敲起代码来不仅不会有任何代码提示,也有可能造成一些比较低级的运行时错误。差评!

基于React.cloneElement()的改造方案

为了改造上面的这种方案,我翻到了一个比较冷门但好用的函数—— React.cloneElement()。这个函数没有很多需要值得注意的点,主要看一眼它的三个入参,第一个是parent element,第二个是parent props,第三个是剩余参数...children,除第一个参数外,其他都是可选值。

举个例子:

<!-- 调用函数 -->
React.cloneElement(<div/>,{},<span/>);
<!-- 相当于创建了这样一个结构 -->
<div> 
    <span></span>
</div>

那么下面开始改造,reduceRight的架子不动,改一下入参的类型和reduceRight的回调。

// ContextComposer.tsx
import React from 'react';
type IContextComposerProps = {
  contexts: React.ReactElement[];
};
const ContextComposer: React.FC<IContextComposerProps> = ({ contexts, children }) => {
  return (
    <>
      {contexts.reduceRight((child, parent) => {
        return React.cloneElement(parent,{},child);
      }, children)}
    </>
  );
};
export default ContextComposer;
// App.tsx
<ContextComposer
  contexts={[
      <ThemeContext.Provider value={"light"} />,
      <UserDataContext.Provider value={{ name: "ascodelife", age: 25 }} />,
      <LoadingContext.Provider value={{ loading, setLoading }} />,
  ]}>
    { children }
</ContextComposer>

经过改造后,我们在传参时就好像是真的在创建一个组件(当然实际上也创建了组件,只是这个组件本身没有被渲染到虚拟Dom上,实际渲染上去的是被克隆后的副本)。同时,我们刚才关注的value的静态类型检查问题也得到了解决。

tips: React.cloneElement(parent,{},child)等价于React.cloneElement(parent,{children:child}),你知道为什么吗?

相关资源

源码已经同步到了github(https://github.com/ascodelife/react-context-provider-composer)。

同时也打包到了npm仓库中(https://www.npmjs.com/package/@ascodelife/react-context-provider-composer),欢迎体验。

以上就是大佬封装React Context Composer的详细步骤(分享)的详细内容,更多请关注站长家园其它相关文章!

本文标签:  React Context Composer

转载请注明来源:大佬封装React Context Composer的详细步骤(分享)

本文永久链接地址:https://www.adminjie.com/post/6753.html

免责声明:
本站所发布的一切资源仅限用于学习和研究目的;不得将上述内容用于商业或者非法用途,否则,一切后果请用户自负。本站信息来自网络,版权争议与本站无关。您必须在下载后的24个小时之内,从您的电脑中彻底删除上述内容。如果您喜欢该程序,请支持正版软件,购买注册,得到更好的正版服务。

附:
二○○二年一月一日《计算机软件保护条例》第十七条规定:为了学习和研究软件内含的设计思想和原理,通过安装、显示、传输或者存储软件等方式使用软件的,可以不经软件著作权人许可,不向其支付报酬!鉴于此,也希望大家按此说明研究软件!

版权声明:
一、本站致力于为软件爱好者提供国内外软件开发技术和软件共享,着力为用户提供优资资源。
二、本站提供的部分源码下载文件为网络共享资源,请于下载后的24小时内删除。如需体验更多乐趣,还请支持正版。
三、我站提供用户下载的所有内容均转自互联网。如有内容侵犯您的版权或其他利益的,若有侵犯你的权益请:提交版权证明文件到邮箱 2225329873#qq.com(#换为@) 站长会进行审查之后,情况属实的会在三个工作日内为您删除。

  • 站长家园(原代码之家)会员升级
  • 最新文章
    • mysql怎样in查询操作排序

      mysql怎样in查询操作排序

      在mysql中,可利用“ORDERBY”子句配合SELECT语句in查询来操作排序,语法为“select*from表名where字段值in(排序1,排...

    • docker能用yum命令吗

      docker能用yum命令吗

      docker能用yum命令,docker容器内安装yum的方法:1、通过“apt-getupdate”更新apt-get指令;2、通过“apt-getins...

    • 如何解决docker telnet不通的问题

      如何解决docker telnet不通的问题

      dockertelnet不通的解决办法:1、查看配置文件;2、修改nginx配置文件的端口为80;3、重新加载即可。本文操作环境:centOS6.8系统、Do...

    • css怎样改变一个图片高度和宽度

      css怎样改变一个图片高度和宽度

      在css中,可以使用width和height属性来改变一个图片高度和宽度,只需要给图片元素添加“width:宽度值;height:高度值;”样式即可。width...

    • docker不能删除镜像怎么办

      docker不能删除镜像怎么办

      docker不能删除镜像的解决办法:1、删除REPOSITORY;2、删除IMAGEID;3、通过“dockerimages”查看镜像即可。本文操作环境:c...

    热门文章