React高阶组件怎么使用

其他教程   发布日期:2025年02月26日   浏览次数:160

这篇文章主要介绍了React高阶组件怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇React高阶组件怎么使用文章都会有所收获,下面我们一起来看看吧。

监介

高阶组件也就是我们常说的HOC,是React中用于复用组件逻辑的一种高级技巧。HOC自身不是React API的一部分,他是一种基于React的组合特性而形成的设计模式。

其实就是组件作为参数,返回值也是组件的函数,他是纯函数,不会修改传入的组件,也不会使用继承来复制其行为。相反,HOC通过将组件包装在容器组件中来组成新组件。HOC是纯函数,没有副作用。

使用HOC的原因

  • 抽取重复代码,实现组件复用:相同功能组件复用

  • 条件渲染,控制组件的渲染逻辑(渲染劫持):权限控制

  • 捕获/劫持被处理组件的生命周期,常见场景:组件渲染性能追踪、日志打点。

HOC的实现方式

先来看一个简单的HOC实现方式

  1. function withLog(Component) {
  2. return function (props) {
  3. console.log(`[${new Date().toISOString()}] ${Component.name} props:`, props);
  4. return <Component {...props} />;
  5. };
  6. }

这个HOC接收一个组件作为参数,返回一个新的组件,新的组件在渲染之前会先输出当前时间和传入的组件的名称和props到控制台,然后再将props传递给原始组件进行渲染。

使用这个HOC可以像下面这样

  1. function MyComponent(props) {
  2. return <div>{props.message}</div>;
  3. }
  4. const MyComponentWithLog = withLog(MyComponent);
  5. ReactDOM.render(
  6. <MyComponentWithLog message="Hello, world!" />,
  7. document.getElementById('root')
  8. );

渲染的时候控制台会输出

[2023-04-16T00:00:00.000Z] MyComponent props: { message: "Hello, world!" }

可以看到,HOC 可以用来增强组件的功能,比如添加日志、添加权限校验、添加数据预处理等。

其实HOC的实现方式主要分为两种:属性代理和反向继承

属性代理

使用组合的方式,将组件包装在容器上,依赖父子组件的生命周期关系来:

  • 返回无状态的函数组件

  • 返回class组件

操作props

  1. // 可以通过属性代理,拦截父组件传递过来的porps并进行处理。
  2. // 返回一个无状态的函数组件
  3. function HOC(WrappedComponent) {
  4. const newProps = { type: 'HOC' };
  5. return props => <WrappedComponent {...props} {...newProps}/>;
  6. }
  7. // 返回一个有状态的 class 组件
  8. function HOC(WrappedComponent) {
  9. return class extends React.Component {
  10. render() {
  11. const newProps = { type: 'HOC' };
  12. return <WrappedComponent {...this.props} {...newProps}/>;
  13. }
  14. };
  15. }

抽象state

通过属性代理无法直接操作原组件的state,可以通过props和回调函数抽象state

  1. function HOC(WrappedComponent) {
  2. return class extends React.Component {
  3. constructor(props) {
  4. super(props);
  5. this.state = {
  6. name: '',
  7. };
  8. this.onChange = this.onChange.bind(this);
  9. }
  10. onChange = (event) => {
  11. this.setState({
  12. name: event.target.value,
  13. })
  14. }
  15. render() {
  16. const newProps = {
  17. name: {
  18. value: this.state.name,
  19. onChange: this.onChange,
  20. },
  21. };
  22. return <WrappedComponent {...this.props} {...newProps} />;
  23. }
  24. };
  25. }
  26. // 使用
  27. @HOC
  28. class Example extends Component {
  29. render() {
  30. return <input name="name" {...this.props.name} />;
  31. }
  32. }

这里我们稍微做一下解释。我们主要的目的是想通过传给HOC的这个组件来修改HOC的state,在HOC中定义了onChange方法目的是可以修改state。如果在WrappedComponent中调用onChange方法呢?我们可以定义一个对象作为WrappedComponent的props,然后将onChange方法存储在定义的对象上,传给WrappedComponent。

在WrappedComponent中,我们定义了一个input,我们都知道input有value属性作为input的值,还有一个onChange方法作为change事件,我们把传给WrappedComponent的props结构刚好就是value和onChange。这样当input在change的时候就可以就修改state的值了。

通过props实现条件渲染

通过props来控制是否渲染及传入数据

  1. import * as React from 'react';
  2. function HOC (WrappedComponent) {
  3. return (props) => (
  4. <div>
  5. {
  6. props.isShow ? (
  7. <WrappedComponent
  8. {...props}
  9. />
  10. ) : <div>暂无数据</div>
  11. }
  12. </div>
  13. );
  14. }
  15. export default HOC;

其他元素wrapper传入的组件

  1. function withBackgroundColor(WrappedComponent) {
  2. return class extends React.Component {
  3. render() {
  4. return (
  5. <div style={{ backgroundColor: '#ccc' }}>
  6. <WrappedComponent {...this.props} {...newProps} />
  7. </div>
  8. );
  9. }
  10. };
  11. }

反向继承

使用一个函数接收一个组件作为参数传入,并返回一个继承了该传入组件的类组件,且在返回组件的

  1. render()
方法中返回了
  1. super.render()
方法。
  1. const HOC = (WrappedComponent) => {
  2. return class extends WrappedComponent {
  3. render() {
  4. return super.render()
  5. }
  6. }
  7. }
  • 允许HOC通过this访问到原组件,可以直接读取和操作原组件的state/ref等

  • 可以通过

    1. render.super()
    获取传入组件的render,可以有选择的渲染劫持
  • 劫持原组件生命周期方法

  1. function HOC(WrappedComponent){
  2. const didMount = WrappedComponent.prototype.componentDidMount;
  3. // 继承了传入组件
  4. return class HOC extends WrappedComponent {
  5. async componentDidMount(){
  6. // 劫持 WrappedComponent 组件的生命周期
  7. if (didMount) {
  8. await didMount.apply(this);
  9. }
  10. ...
  11. }
  12. render(){
  13. //使用 super 调用传入组件的 render 方法
  14. return super.render();
  15. }
  16. }
  17. }

读取/操作原组件的state

  1. function HOC(WrappedComponent){
  2. const didMount = WrappedComponent.prototype.componentDidMount;
  3. // 继承了传入组件
  4. return class HOC extends WrappedComponent {
  5. async componentDidMount(){
  6. if (didMount) {
  7. await didMount.apply(this);
  8. }
  9. // 将 state 中的 number 值修改成 2
  10. this.setState({ number: 2 });
  11. }
  12. render(){
  13. //使用 super 调用传入组件的 render 方法
  14. return super.render();
  15. }
  16. }
  17. }

通过

  1. this.setState({ number: 2 });
修改了原组件
  1. WrappedComponent
中的state的number值

条件渲染

  1. const HOC = (WrappedComponent) =>
  2. class extends WrappedComponent {
  3. render() {
  4. if (this.props.isRender) {
  5. return super.render();
  6. } else {
  7. return <div>暂无数据</div>;
  8. }
  9. }
  10. }

修改react树

  1. // 修改返回render结果
  2. function HigherOrderComponent(WrappedComponent) {
  3. return class extends WrappedComponent {
  4. render() {
  5. const tree = super.render();
  6. const newProps = {};
  7. if (tree && tree.type === 'input') {
  8. newProps.value = 'something here';
  9. }
  10. const props = {
  11. ...tree.props,
  12. ...newProps,
  13. };
  14. const newTree = React.cloneElement(tree, props, tree.props.children);
  15. return newTree;
  16. }
  17. };
  18. }

这里首先通过

  1. super.render()
拿到需要渲染的树,然后对这个渲染树做了修改。比如如果是一个input,则修改它的value值。

属性代理和反向继承对比

  • 属性代理:从"组合"角度出发,有利于从外部操作WrappedComponent,可以操作props,或者在WrappedComponent外加一些拦截器(如:条件渲染,增加外部样式)

  • 反向继承:从"继承"角度出发,从内部操作WrappedComponent,可以操作组件内部的state,生命周期和render等,功能更强大

以上就是React高阶组件怎么使用的详细内容,更多关于React高阶组件怎么使用的资料请关注九品源码其它相关文章!