详解React中共享组件逻辑的三种方式
废话少说,这三种方式分别是:renderprops、高阶组件和自定义Hook。下面依次演示
假设有一个TimeOnPage组件专门用来记录用户在当前页面停留时间,像这样:
constTimeOnPage=()=>{ const[second,setSecond]=useState(0); useEffect(()=>{ setTimeout(()=>{ setSecond(second+1); },1000); },[second]); return(停留时间:{second}秒); }
如果另一个组件需要复用这个功能,我们能否封装一下,以便轻松地与其它组件共享?
一般很自然地想到子组件嵌套的方式,利用props传参
constChild=(props)=>{ returnstayTime:{props.stayTime}s; }; constTimeOnPage=()=>{ const[second,setSecond]=useState(0); useEffect(()=>{ setTimeout(()=>{ setSecond(second+1); },1000); },[second]); return(); }
这属于在TimeOnPage组件内部硬编码,还没有达到封装复用的目标。看看renderprops怎么做?
renderprops
“renderprop”是指一种在React组件之间使用一个值为函数的prop共享代码的简单技术
接上文,在TimeOnPage里定义一个值为函数的prop,想渲染什么组件,在函数里返回即可,函数的参数就是想要共享的state。
constChild=(props)=>{ returnstayTime:{props.stayTime}s; }; constTimeOnPage=(props)=>{ const[second,setSecond]=useState(0); useEffect(()=>{ setTimeout(()=>{ setSecond(second+1); },1000); },[second]); return{props.render(second)}; };
其实,renderprop就是一个用于告知组件需要渲染什么内容的函数prop。
ReactRouter也用到了这项技术。
Home}/>
高阶组件
高阶组件(HOC)是React中用于复用组件逻辑的一种高级技巧。HOC自身不是ReactAPI的一部分,它是一种基于React的组合特性而形成的设计模式。
高阶组件是一个函数,参数是一个需要被复用的组件A,返回值是一个新的组件N。新组件N是在组件A的基础上做了一些加工,但不会修改组件A本身,只是功能增强。
假设有一个新闻列表组件长这样:
constNewList=()=>{ return(); }
- newsitem
- newsitem
想要在新闻列表加载期间显示loading动画组件
constLoading=()=>{ //loading动画 } constNewList=({isLoading})=>{ returnisLoading?():( ); };
- newsitem
- newsitem
假设现在Table组件也要在加载数据期间显示loading动画组件,遵循类似的模式
constLoading=()=>{ //loading动画 } constDataList=({isLoading,...props})=>{ returnisLoading?():(
以上,你会发现DataList和NewList结构极度相似,如果还有第三个、第四个组件要加loading,继续照这个模式重复第三次、第四次吗?这不是最理想的做法,更好的做法是,使用高阶组件把这个模式抽象出来:
constWithLoading=(WrappedComponent)=>{ return({isLoading,...props})=>{ returnisLoading?: ; } };
然后就可以在不修改NewList和DataList的情况下分别给他们增加loading
constNewList=()=>{ return(); }; constDataList=(props)=>{ return
- newsitem
- newsitem
自定义Hook
Hook是React16.8的新增特性。它可以让你在不编写class的情况下使用state以及其他的React特性。
ReactHook有useState、useEffect等,它们都是函数,自定义Hook也是一个函数,它的名称同样以use开头,函数内部可以调用其它Hook。与React组件不同的是,自定义Hook可以没有返回值。与普通函数不同的是,自定义Hook内部可以调用其它Hook,而普通函数则不行。
在写业务逻辑过程中,一般会将一些可重用的的方法定义成工具函数,然后就可以到处复用。同样,通过自定义Hook,可以将组件逻辑提取到可重用的函数中。到底选择自定义Hook还是工具函数,取决于要提取的组件逻辑需不需要用到其他Hook,如果需要,就选择自定义Hook,否则用工具函数即可。
回到本文第一个TimeOnPage组件,改成自定义Hook的形式
constuseTimeOnPage=()=>{ const[second,setSecond]=useState(0); useEffect(()=>{ setTimeout(()=>{ setSecond(second+1); },1000); },[second]); returnsecond; }
使用方法
constDemo=()=>{ conststayTime=useTimeOnPage(); return当前页面停留时间:{stayTime}秒}
总结
三种共享组件逻辑的方式有各自的适用场景:
renderprops适合共享那些有不同子组件/子元素的父组件,子组件/子元素的“坑位”已经定义好了,只能渲染在指定位置;
高阶组件适合在不修改原有组件的基础上对组件进行扩展;
自定义Hook能做的,纯函数基本上也能做,只是有时候用自定义Hook实现会更方便快捷。
本文链接:Github
到此这篇关于详解React中共享组件逻辑的三种方式的文章就介绍到这了,更多相关React共享组件逻辑内容请搜索毛票票以前的文章或继续浏览下面的相关文章希望大家以后多多支持毛票票!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。