【React】常见的 HOC 使用案例

高阶组件(Higher-Order Component,HOC)是一种用于在 React 中复用组件逻辑的技术。以下是几个常见的 HOC 使用案例,以及详细的代码示例。

1. 日志记录 HOC

这个高阶组件将在每次组件更新时记录日志。

LoggingHOC.js
js 复制代码
import React from 'react';

const withLogging = (WrappedComponent) => {
  return class extends React.Component {
    componentDidMount() {
      console.log(`${WrappedComponent.name} mounted`);
    }

    componentDidUpdate() {
      console.log(`${WrappedComponent.name} updated`);
    }

    componentWillUnmount() {
      console.log(`${WrappedComponent.name} will unmount`);
    }

    render() {
      return <WrappedComponent {...this.props} />;
    }
  };
};

export default withLogging;
使用日志记录 HOC
js 复制代码
// src/App.js
import React from 'react';
import withLogging from './LoggingHOC';

const MyComponent = () => {
  return <div>My Component</div>;
};

const LoggedMyComponent = withLogging(MyComponent);

const App = () => {
  return (
    <div>
      <LoggedMyComponent />
    </div>
  );
};

export default App;

2. 数据获取 HOC

这个高阶组件在组件挂载时从一个 API 获取数据,并将数据传递给被包装的组件。

FetchDataHOC.js
js 复制代码
import React from 'react';

const withFetchData = (url) => (WrappedComponent) => {
  return class extends React.Component {
    state = {
      data: null,
      loading: true,
      error: null,
    };

    async componentDidMount() {
      try {
        const response = await fetch(url);
        const data = await response.json();
        this.setState({ data, loading: false });
      } catch (error) {
        this.setState({ error, loading: false });
      }
    }

    render() {
      const { data, loading, error } = this.state;
      return <WrappedComponent data={data} loading={loading} error={error} {...this.props} />;
    }
  };
};

export default withFetchData;
使用数据获取 HOC
js 复制代码
// src/App.js
import React from 'react';
import withFetchData from './FetchDataHOC';

const DataComponent = ({ data, loading, error }) => {
  if (loading) return <div>Loading...</div>;
  if (error) return <div>Error: {error.message}</div>;
  return <div>Data: {JSON.stringify(data)}</div>;
};

const FetchDataComponent = withFetchData('https://api.example.com/data')(DataComponent);

const App = () => {
  return (
    <div>
      <FetchDataComponent />
    </div>
  );
};

export default App;

3. 权限控制 HOC

这个高阶组件根据用户权限来控制组件的渲染。

withAuthorization.js
js 复制代码
import React from 'react';

const withAuthorization = (requiredRole) => (WrappedComponent) => {
  return class extends React.Component {
    render() {
      const { user } = this.props;
      if (user.role !== requiredRole) {
        return <div>You do not have permission to view this page</div>;
      }
      return <WrappedComponent {...this.props} />;
    }
  };
};

export default withAuthorization;
使用权限控制 HOC
js 复制代码
// src/App.js
import React from 'react';
import withAuthorization from './withAuthorization';

const AdminPage = () => {
  return <div>Admin Page</div>;
};

const AuthorizedAdminPage = withAuthorization('admin')(AdminPage);

const App = () => {
  const user = { role: 'user' }; // change to 'admin' to see the page
  return (
    <div>
      <AuthorizedAdminPage user={user} />
    </div>
  );
};

export default App;

4. 动态样式 HOC

这个高阶组件根据 props 动态添加样式。

withDynamicStyles.js
js 复制代码
import React from 'react';

const withDynamicStyles = (WrappedComponent) => {
  return class extends React.Component {
    render() {
      const style = {
        color: this.props.color || 'black',
        fontSize: this.props.fontSize || '16px',
      };
      return <WrappedComponent {...this.props} style={style} />;
    }
  };
};

export default withDynamicStyles;
使用动态样式 HOC
js 复制代码
// src/App.js
import React from 'react';
import withDynamicStyles from './withDynamicStyles';

const StyledComponent = ({ style }) => {
  return <div style={style}>Styled Component</div>;
};

const DynamicStyledComponent = withDynamicStyles(StyledComponent);

const App = () => {
  return (
    <div>
      <DynamicStyledComponent color="red" fontSize="20px" />
    </div>
  );
};

export default App;

总结

高阶组件是一种强大的模式,可以在 React 中实现代码复用和逻辑抽象。通过高阶组件,你可以:

  • 提取和重用跨组件的逻辑
  • 控制组件的渲染
  • 操作传递给组件的 props
  • 管理和注入状态
相关推荐
吴声子夜歌15 分钟前
ES6——Iterator和for...of循环详解
前端·javascript·es6
小李子呢021119 分钟前
前端八股3---ref和reactive
开发语言·前端·javascript
落魄江湖行23 分钟前
基础篇三 Nuxt4 组件进阶:插槽与事件传递
前端·nuxt4
kerli24 分钟前
Compose 组件:LazyColumn 核心参数与 key/contentType 详解
android·前端
好运的阿财24 分钟前
“锟斤拷”问题——程序中用powershell执行命令出现中文乱码的解决办法
linux·前端·人工智能·机器学习·架构·编辑器·vim
踩着两条虫36 分钟前
VTJ.PRO AI + 低代码实战:接入高德地图
前端·vue.js·ai编程
绝世唐门三哥36 分钟前
React性能优化:memo、useMemo和useCallback全解析
前端·react.js·memo
兔子零102439 分钟前
Claude Code 都把宠物养进终端了,我做了一个真正能长期玩的中文宠物游戏
前端·游戏·游戏开发
xiaotao13139 分钟前
Vite 与 Webpack 开发/打包时环境变量对比
前端·vue.js·webpack
摆烂工程师44 分钟前
教你如何查询 Codex 最新额度是多少,以及 ChatGPT Pro、Plus、Business 最新额度变化
前端·后端·ai编程