详解React中共享组件逻辑的三种方式


Posted in Javascript onFebruary 02, 2021

废话少说,这三种方式分别是:render props、高阶组件和自定义Hook。下面依次演示

假设有一个TimeOnPage组件专门用来记录用户在当前页面停留时间,像这样:

const TimeOnPage = () => {
 const [second, setSecond] = useState(0);

 useEffect(() => {
  setTimeout(() => {
   setSecond(second + 1);
  }, 1000);
 }, [second]);
 return (
  <div>停留时间:{second}秒</div>
 );
}

如果另一个组件需要复用这个功能,我们能否封装一下,以便轻松地与其它组件共享?

一般很自然地想到子组件嵌套的方式,利用props传参

const Child = (props) => {
 return <div>stayTime: {props.stayTime}s</div>;
};

const TimeOnPage = () => {
 const [second, setSecond] = useState(0);

 useEffect(() => {
  setTimeout(() => {
   setSecond(second + 1);
  }, 1000);
 }, [second]);
 return (
  <div>
   <Child stayTime={second} />
  </div>
 );
}

这属于在 TimeOnPage组件内部硬编码,还没有达到封装复用的目标。看看render props怎么做?

render props

“render prop” 是指一种在 React 组件之间使用一个值为函数的 prop 共享代码的简单技术

接上文,在TimeOnPage里定义一个值为函数的prop,想渲染什么组件,在函数里返回即可,函数的参数就是想要共享的state。

const Child = (props) => {
 return <div>stayTime: {props.stayTime}s</div>;
};

const TimeOnPage = (props) => {
 const [second, setSecond] = useState(0);

 useEffect(() => {
  setTimeout(() => {
   setSecond(second + 1);
  }, 1000);
 }, [second]);
 return <div>{props.render(second)}</div>;
};

<TimeOnPage render={(stayTime) => <Child stayTime={stayTime} />

其实,render prop 就是一个用于告知组件需要渲染什么内容的函数prop。
React Router也用到了这项技术。

<Router>
 <Route path="/home" render={() => <div>Home</div>} />
</Router>

高阶组件

高阶组件(HOC)是 React 中用于复用组件逻辑的一种高级技巧。HOC 自身不是 React API 的一部分,它是一种基于 React 的组合特性而形成的设计模式。

高阶组件是一个函数,参数是一个需要被复用的组件A,返回值是一个新的组件N。新组件N是在组件A的基础上做了一些加工,但不会修改组件A本身,只是功能增强。

假设有一个新闻列表组件长这样:

const NewList = () => {
 return (
  <div>
   <ul>
    <li>news item</li>
    <li>news item</li>
   </ul>
  </div>
 );
}

想要在新闻列表加载期间显示loading动画组件 <Loading />,通常会这么做

const Loading = () => {
 // loading动画
}
const NewList = ({ isLoading }) => {
 return isLoading ? (
  <Loading />
 ) : (
  <div>
   <ul>
    <li>news item</li>
    <li>news item</li>
   </ul>
  </div>
 );
};

假设现在Table组件也要在加载数据期间显示loading动画组件,遵循类似的模式

const Loading = () => {
 // loading动画
}
const DataList = ({ isLoading, ...props }) => {
 return isLoading ? (
  <Loading />
 ) : (
  <Table {...props} />
 );
};

以上,你会发现DataList和NewList结构极度相似,如果还有第三个、第四个组件要加loading,继续照这个模式重复第三次、第四次吗?这不是最理想的做法,更好的做法是,使用高阶组件把这个模式抽象出来:

const WithLoading = (WrappedComponent) => {
 return ({isLoading, ...props}) => {
  return isLoading ? <Loading /> : <WrappedComponent {...props} />;
 }
};

然后就可以在不修改NewList和DataList的情况下分别给他们增加loading

const NewList = () => {
 return (
  <div>
   <ul>
    <li>news item</li>
    <li>news item</li>
   </ul>
  </div>
 );
};

const DataList = (props) => {
 return <Table {...props} />
};

const WithLoading = (WrappedComponent) => {
 return ({isLoading, ...props}) => {
  return isLoading ? <Loading /> : <WrappedComponent {...props} />;
 }
};
// 带loading的NewList
const WithLoadingNewList = WithLoading(<NewList />)
// 带loading的DataList
const WithLoadingDataList = WithLoading(<DataList />)

自定义Hook

Hook 是 React 16.8 的新增特性。它可以让你在不编写 class 的情况下使用 state 以及其他的 React 特性。

React Hook有useState、useEffect等,它们都是函数,自定义Hook也是一个函数,它的名称同样以use开头,函数内部可以调用其它Hook。与React组件不同的是,自定义Hook可以没有返回值。与普通函数不同的是,自定义Hook内部可以调用其它Hook,而普通函数则不行。

在写业务逻辑过程中,一般会将一些可重用的的方法定义成工具函数,然后就可以到处复用。同样,通过自定义 Hook,可以将组件逻辑提取到可重用的函数中。到底选择自定义Hook还是工具函数,取决于要提取的组件逻辑需不需要用到其他Hook,如果需要,就选择自定义Hook,否则用工具函数即可。

回到本文第一个 TimeOnPage组件,改成自定义Hook的形式

const useTimeOnPage = () => {
 const [second, setSecond] = useState(0);

 useEffect(() => {
  setTimeout(() => {
   setSecond(second + 1);
  }, 1000);
 }, [second]);
 return second;
}

使用方法

const Demo = () => {
 const stayTime = useTimeOnPage();
 return <div>当前页面停留时间:{stayTime}秒</div>
}

总结

三种共享组件逻辑的方式有各自的适用场景:
render props适合共享那些有不同子组件/子元素的父组件,子组件/子元素的“坑位”已经定义好了,只能渲染在指定位置;
高阶组件适合在不修改原有组件的基础上对组件进行扩展;
自定义Hook能做的,纯函数基本上也能做,只是有时候用自定义Hook实现会更方便快捷。
本文链接:Github

 到此这篇关于详解React中共享组件逻辑的三种方式的文章就介绍到这了,更多相关React 共享组件逻辑内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
iframe自适应宽度、高度 ie6 7 8,firefox 3.86下测试通过
Jul 29 Javascript
jQuery动态添加的元素绑定事件处理函数代码
Aug 02 Javascript
Javascript 按位左移运算符使用介绍(
Feb 04 Javascript
js操作DOM--添加、删除节点的简单实例
Jul 08 Javascript
无法获取隐藏元素宽度和高度的解决方案
Mar 07 Javascript
Vue.js在数组中插入重复数据的实现代码
Nov 17 Javascript
Vue使用vue-area-linkage实现地址三级联动效果的示例
Jun 27 Javascript
浅谈react性能优化的方法
Sep 05 Javascript
JointJS流程图的绘制方法
Dec 03 Javascript
微信小程序实现发微博功能的示例代码
Jun 24 Javascript
Element Dropdown下拉菜单的使用方法
Jul 26 Javascript
vue添加自定义右键菜单的完整实例
Dec 08 Vue.js
详解微信小程序轨迹回放实现及遇到的坑
Feb 02 #Javascript
学习 Vue.js 遇到的那些坑
Feb 02 #Vue.js
Vue常用API、高级API的相关总结
Feb 02 #Vue.js
JavaScript实现点击自制菜单效果
Feb 02 #Javascript
Vue项目打包部署到apache服务器的方法步骤
Feb 01 #Vue.js
如何在vue中使用video.js播放m3u8格式的视频
Feb 01 #Vue.js
Vue 实现可视化拖拽页面编辑器
Feb 01 #Vue.js
You might like
神族 Protoss 历史背景
2020/03/14 星际争霸
解析php中的fopen()函数用打开文件模式说明
2013/06/20 PHP
php中文乱码怎么办如何让浏览器自动识别utf-8
2014/01/15 PHP
destoon实现不同会员组公司名称显示不同的颜色的方法
2014/08/22 PHP
调试WordPress中定时任务的相关PHP脚本示例
2015/12/10 PHP
Yii2创建表单(ActiveForm)方法详解
2016/07/23 PHP
PHP基于堆栈实现的高级计算器功能示例
2017/09/15 PHP
ext jquery 简单比较
2010/04/07 Javascript
基于jquery点击自以外任意处,关闭自身的代码
2012/02/10 Javascript
javascript学习笔记(十六) 系统对话框(alert、confirm、prompt)
2012/06/20 Javascript
Extjs显示从数据库取出时间转换JSON后的出现问题
2012/11/20 Javascript
深入探寻javascript定时器
2015/01/02 Javascript
基于JavaScript实现div层跟随滚动条滑动
2016/01/12 Javascript
jquery处理checkbox(复选框)是否被选中实例代码
2017/06/12 jQuery
jquery animate动画持续运动的实例
2017/11/29 jQuery
vue实现带复选框的树形菜单
2019/05/27 Javascript
浅析vue中的nextTick
2020/12/28 Vue.js
pytorch下使用LSTM神经网络写诗实例
2020/01/14 Python
pycharm 代码自动补全的实现方法(图文)
2020/09/18 Python
web字体加载方案优化小结
2019/11/29 HTML / CSS
澳大利亚厨房和家用电器购物网站:Bing Lee
2021/01/11 全球购物
诺思信科技(南京)有限公司.NET笔试题答案
2013/07/06 面试题
《月球之谜》教学反思
2014/04/10 职场文书
项目经理任命书范本
2014/06/05 职场文书
应聘会计求职信
2014/06/11 职场文书
计算机系统管理员求职信
2014/06/20 职场文书
人身意外保险授权委托书
2014/10/01 职场文书
2014年标准化工作总结
2014/12/17 职场文书
六一儿童节园长致辞
2015/07/31 职场文书
任命书格式模板
2015/09/22 职场文书
2016年主题党日活动总结
2016/04/05 职场文书
AJAX学习笔记
2021/05/18 Javascript
浅谈MySQL之浅入深出页原理
2021/06/23 MySQL
java设计模式--建造者模式详解
2021/07/21 Java/Android
阿里云服务器Ubuntu 20.04上安装Odoo 15
2022/05/20 Servers
Vue深入理解插槽slot的使用
2022/08/05 Vue.js