React中共享组件逻辑方式有什么,分别是怎样的
Admin 2022-06-01 群英技术资讯 266 次浏览
废话少说,这三种方式分别是: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 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 是 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实现会更方便快捷。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。
猜你喜欢
对axios进行封装以及将API接口按业务模块统一管理,有助于我们简化代码,方便后期维护。本文介绍了如何用vue封装axios请求,感兴趣的同学,可以参考下。
这篇文章给大家分享的是用jQuery写一个查看图片的功能的内容,可以实现点击左右查看图片效果等等,小编觉得挺实用的,因此分享给大家做个参考,接下来一起跟随小编看看吧。
utf8编码算法是什么样的,如何实现?小编这就来给大家分析分析下,对此感兴趣的话可以看看以下知识点,有需要的朋友一定不要错过了。
这篇文章主要介绍了关于vue中的时间格式转化问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
今天给大家分享的是用node实现设置静态文件缓存的方法,对于缓存的内容是比较基础的,也是需要掌握的内容,因此下文就给大家来介绍一下node实现静态文件缓存,有需要的朋友可以参考。
成为群英会员,开启智能安全云计算之旅
立即注册Copyright © QY Network Company Ltd. All Rights Reserved. 2003-2020 群英 版权所有
增值电信经营许可证 : B1.B2-20140078 粤ICP备09006778号 域名注册商资质 粤 D3.1-20240008