欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

React用法之高階組件的用法詳解

 更新時(shí)間:2023年04月17日 11:34:19   作者:翰玥  
高階組件也就是我們常說的HOC,是React中用于復(fù)用組件邏輯的一種高級(jí)技巧。這篇文章主要通過一些示例帶大家學(xué)習(xí)一下高階組件的使用,希望對(duì)大家有所幫助

監(jiān)介

高階組件也就是我們常說的HOC,是React中用于復(fù)用組件邏輯的一種高級(jí)技巧。HOC自身不是React API的一部分,他是一種基于React的組合特性而形成的設(shè)計(jì)模式。

其實(shí)就是組件作為參數(shù),返回值也是組件的函數(shù),他是純函數(shù),不會(huì)修改傳入的組件,也不會(huì)使用繼承來復(fù)制其行為。相反,HOC通過將組件包裝在容器組件中來組成新組件。HOC是純函數(shù),沒有副作用。

使用HOC的原因

  • 抽取重復(fù)代碼,實(shí)現(xiàn)組件復(fù)用:相同功能組件復(fù)用
  • 條件渲染,控制組件的渲染邏輯(渲染劫持):權(quán)限控制
  • 捕獲/劫持被處理組件的生命周期,常見場(chǎng)景:組件渲染性能追蹤、日志打點(diǎn)。

HOC的實(shí)現(xiàn)方式

先來看一個(gè)簡(jiǎn)單的HOC實(shí)現(xiàn)方式

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

這個(gè)HOC接收一個(gè)組件作為參數(shù),返回一個(gè)新的組件,新的組件在渲染之前會(huì)先輸出當(dāng)前時(shí)間和傳入的組件的名稱和props到控制臺(tái),然后再將props傳遞給原始組件進(jìn)行渲染。

使用這個(gè)HOC可以像下面這樣

function MyComponent(props) {
	return <div>{props.message}</div>;
}

const MyComponentWithLog = withLog(MyComponent);

ReactDOM.render(
	<MyComponentWithLog message="Hello, world!" />,
	document.getElementById('root')
);

渲染的時(shí)候控制臺(tái)會(huì)輸出

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

可以看到,HOC 可以用來增強(qiáng)組件的功能,比如添加日志、添加權(quán)限校驗(yàn)、添加數(shù)據(jù)預(yù)處理等。

其實(shí)HOC的實(shí)現(xiàn)方式主要分為兩種:屬性代理和反向繼承

屬性代理

使用組合的方式,將組件包裝在容器上,依賴父子組件的生命周期關(guān)系來:

  • 返回?zé)o狀態(tài)的函數(shù)組件
  • 返回class組件

操作props

// 可以通過屬性代理,攔截父組件傳遞過來的porps并進(jìn)行處理。

// 返回一個(gè)無狀態(tài)的函數(shù)組件
function HOC(WrappedComponent) {
  const newProps = { type: 'HOC' };
  return props => <WrappedComponent {...props} {...newProps}/>;
}

// 返回一個(gè)有狀態(tài)的 class 組件
function HOC(WrappedComponent) {
  return class extends React.Component {
    render() {
      const newProps = { type: 'HOC' };
      return <WrappedComponent {...this.props} {...newProps}/>;
    }
  };
}

抽象state

通過屬性代理無法直接操作原組件的state,可以通過props和回調(diào)函數(shù)抽象state

function HOC(WrappedComponent) {
  return class extends React.Component {
    constructor(props) {
      super(props);
      this.state = {
        name: '',
      };
      this.onChange = this.onChange.bind(this);
    }
    
    onChange = (event) => {
      this.setState({
        name: event.target.value,
      })
    }
    
    render() {
      const newProps = {
        name: {
          value: this.state.name,
          onChange: this.onChange,
        },
      };
      return <WrappedComponent {...this.props} {...newProps} />;
    }
  };
}

// 使用
@HOC
class Example extends Component {
  render() {
    return <input name="name" {...this.props.name} />;
  }
}

這里我們稍微做一下解釋。我們主要的目的是想通過傳給HOC的這個(gè)組件來修改HOC的state,在HOC中定義了onChange方法目的是可以修改state。如果在WrappedComponent中調(diào)用onChange方法呢?我們可以定義一個(gè)對(duì)象作為WrappedComponent的props,然后將onChange方法存儲(chǔ)在定義的對(duì)象上,傳給WrappedComponent。

在WrappedComponent中,我們定義了一個(gè)input,我們都知道input有value屬性作為input的值,還有一個(gè)onChange方法作為change事件,我們把傳給WrappedComponent的props結(jié)構(gòu)剛好就是value和onChange。這樣當(dāng)input在change的時(shí)候就可以就修改state的值了。

通過props實(shí)現(xiàn)條件渲染

通過props來控制是否渲染及傳入數(shù)據(jù)

import * as React from 'react';

function HOC (WrappedComponent) {
  return (props) => (
  <div>
    {
      props.isShow ? (
        <WrappedComponent
          {...props}
        />
      ) : <div>暫無數(shù)據(jù)</div>
    }
  </div>
  );
}

export default HOC;

其他元素wrapper傳入的組件

function withBackgroundColor(WrappedComponent) {
  return class extends React.Component {
    render() {
      return (
        <div style={{ backgroundColor: '#ccc' }}>
            <WrappedComponent {...this.props} {...newProps} />
        </div>
      );
    }
  };
}

反向繼承

使用一個(gè)函數(shù)接收一個(gè)組件作為參數(shù)傳入,并返回一個(gè)繼承了該傳入組件的類組件,且在返回組件的render()方法中返回了super.render()方法。

const HOC = (WrappedComponent) => {
	return class extends WrappedComponent {
		render() {
			return super.render()
		}
	}
}
  • 允許HOC通過this訪問到原組件,可以直接讀取和操作原組件的state/ref等
  • 可以通過render.super()獲取傳入組件的render,可以有選擇的渲染劫持
  • 劫持原組件生命周期方法
function HOC(WrappedComponent){
  const didMount = WrappedComponent.prototype.componentDidMount;
  
  // 繼承了傳入組件
  return class HOC extends WrappedComponent {
    async componentDidMount(){
      // 劫持 WrappedComponent 組件的生命周期
      if (didMount) {
        await didMount.apply(this);
      }
      ...
    }

    render(){
      //使用 super 調(diào)用傳入組件的 render 方法
      return super.render();
    }
  }
}

讀取/操作原組件的state

function HOC(WrappedComponent){
  const didMount = WrappedComponent.prototype.componentDidMount;
  // 繼承了傳入組件
  return class HOC extends WrappedComponent {
    async componentDidMount(){
      if (didMount) {
        await didMount.apply(this);
      }
      // 將 state 中的 number 值修改成 2
      this.setState({ number: 2 });
    }

    render(){
      //使用 super 調(diào)用傳入組件的 render 方法
      return super.render();
    }
  }
}

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

條件渲染

const HOC = (WrappedComponent) =>
  class extends WrappedComponent {
    render() {
      if (this.props.isRender) {
        return super.render();
      } else {
        return <div>暫無數(shù)據(jù)</div>;
      }
    }
  }

修改react樹

// 修改返回render結(jié)果
function HigherOrderComponent(WrappedComponent) {
  return class extends WrappedComponent {
    render() {
      const tree = super.render();
      const newProps = {};
      if (tree && tree.type === 'input') {
        newProps.value = 'something here';
      }
      const props = {
        ...tree.props,
        ...newProps,
      };
      const newTree = React.cloneElement(tree, props, tree.props.children);
      return newTree;
    }
  };
}

這里首先通過super.render()拿到需要渲染的樹,然后對(duì)這個(gè)渲染樹做了修改。比如如果是一個(gè)input,則修改它的value值。

屬性代理和反向繼承對(duì)比

  • 屬性代理:從"組合"角度出發(fā),有利于從外部操作WrappedComponent,可以操作props,或者在WrappedComponent外加一些攔截器(如:條件渲染,增加外部樣式)
  • 反向繼承:從"繼承"角度出發(fā),從內(nèi)部操作WrappedComponent,可以操作組件內(nèi)部的state,生命周期和render等,功能更強(qiáng)大

到此這篇關(guān)于React用法之高階組件的用法詳解的文章就介紹到這了,更多相關(guān)React高階組件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • react數(shù)據(jù)管理機(jī)制React.Context源碼解析

    react數(shù)據(jù)管理機(jī)制React.Context源碼解析

    這篇文章主要為大家介紹了react數(shù)據(jù)管理機(jī)制React.Context源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-11-11
  • webpack3+React 的配置全解

    webpack3+React 的配置全解

    本篇文章主要介紹了webpack3+React 的配置全解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-08-08
  • React forwardRef的使用方法及注意點(diǎn)

    React forwardRef的使用方法及注意點(diǎn)

    React.forwardRef的API中ref必須指向dom元素而不是React組件,通過一段示例代碼給大家介紹了React forwardRef使用方法及注意點(diǎn)還有一些特殊情況分析,感興趣的朋友跟隨小編一起看看吧
    2021-06-06
  • React props和state屬性的具體使用方法

    React props和state屬性的具體使用方法

    本篇文章主要介紹了React props和state屬性的具體使用方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-04-04
  • 學(xué)習(xí)ahooks useRequest并實(shí)現(xiàn)手寫

    學(xué)習(xí)ahooks useRequest并實(shí)現(xiàn)手寫

    這篇文章主要為大家介紹了學(xué)習(xí)ahooks useRequest并實(shí)現(xiàn)手寫示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-03-03
  • React中父組件如何獲取子組件的值或方法

    React中父組件如何獲取子組件的值或方法

    這篇文章主要介紹了React中父組件如何獲取子組件的值或方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • 使用react-virtualized實(shí)現(xiàn)圖片動(dòng)態(tài)高度長(zhǎng)列表的問題

    使用react-virtualized實(shí)現(xiàn)圖片動(dòng)態(tài)高度長(zhǎng)列表的問題

    一般我們?cè)趯憆eact項(xiàng)目中,同時(shí)渲染很多dom節(jié)點(diǎn),會(huì)造成頁面卡頓, 空白的情況。為了解決這個(gè)問題,今天小編給大家分享一篇教程關(guān)于react-virtualized實(shí)現(xiàn)圖片動(dòng)態(tài)高度長(zhǎng)列表的問題,感興趣的朋友跟隨小編一起看看吧
    2021-05-05
  • react跳轉(zhuǎn)后路由變了頁面沒刷新的解決

    react跳轉(zhuǎn)后路由變了頁面沒刷新的解決

    這篇文章主要介紹了react跳轉(zhuǎn)后路由變了頁面沒刷新的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • React項(xiàng)目中報(bào)錯(cuò):Parsing error: The keyword 'import' is reservedeslint的問題及解決方法

    React項(xiàng)目中報(bào)錯(cuò):Parsing error: The keyword &a

    ESLint 默認(rèn)使用的是 ES5 語法,如果你想使用 ES6 或者更新的語法,你需要在 ESLint 的配置文件如:.eslintrc.js等中設(shè)置 parserOptions,這篇文章主要介紹了React項(xiàng)目中報(bào)錯(cuò):Parsing error: The keyword 'import' is reservedeslint的問題及解決方法,需要的朋友可以參考下
    2023-12-12
  • React Native自定義路由管理的深入理解

    React Native自定義路由管理的深入理解

    路由管理的功能主要指的頁面跳轉(zhuǎn)、goBack、帶參數(shù)跳轉(zhuǎn)等功能,這篇文章主要給大家介紹了關(guān)于React Native自定義路由管理的相關(guān)資料,需要的朋友可以參考下
    2021-08-08

最新評(píng)論