中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

React高階組件怎么使用

發布時間:2023-04-17 15:02:39 來源:億速云 閱讀:111 作者:iii 欄目:開發技術

這篇文章主要介紹了React高階組件怎么使用的相關知識,內容詳細易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇React高階組件怎么使用文章都會有所收獲,下面我們一起來看看吧。

監介

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

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

使用HOC的原因

  • 抽取重復代碼,實現組件復用:相同功能組件復用

  • 條件渲染,控制組件的渲染邏輯(渲染劫持):權限控制

  • 捕獲/劫持被處理組件的生命周期,常見場景:組件渲染性能追蹤、日志打點。

HOC的實現方式

先來看一個簡單的HOC實現方式

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

這個HOC接收一個組件作為參數,返回一個新的組件,新的組件在渲染之前會先輸出當前時間和傳入的組件的名稱和props到控制臺,然后再將props傳遞給原始組件進行渲染。

使用這個HOC可以像下面這樣

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

const MyComponentWithLog = withLog(MyComponent);

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

渲染的時候控制臺會輸出

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

可以看到,HOC 可以用來增強組件的功能,比如添加日志、添加權限校驗、添加數據預處理等。

其實HOC的實現方式主要分為兩種:屬性代理和反向繼承

屬性代理

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

  • 返回無狀態的函數組件

  • 返回class組件

操作props

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

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

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

抽象state

通過屬性代理無法直接操作原組件的state,可以通過props和回調函數抽象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的這個組件來修改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來控制是否渲染及傳入數據

import * as React from 'react';

function HOC (WrappedComponent) {
  return (props) => (
  <div>
    {
      props.isShow ? (
        <WrappedComponent
          {...props}
        />
      ) : <div>暫無數據</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>
      );
    }
  };
}

反向繼承

使用一個函數接收一個組件作為參數傳入,并返回一個繼承了該傳入組件的類組件,且在返回組件的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 調用傳入組件的 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 調用傳入組件的 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>暫無數據</div>;
      }
    }
  }

修改react樹

// 修改返回render結果
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()拿到需要渲染的樹,然后對這個渲染樹做了修改。比如如果是一個input,則修改它的value值。

屬性代理和反向繼承對比

  • 屬性代理:從"組合"角度出發,有利于從外部操作WrappedComponent,可以操作props,或者在WrappedComponent外加一些攔截器(如:條件渲染,增加外部樣式)

  • 反向繼承:從"繼承"角度出發,從內部操作WrappedComponent,可以操作組件內部的state,生命周期和render等,功能更強大

關于“React高階組件怎么使用”這篇文章的內容就介紹到這里,感謝各位的閱讀!相信大家對“React高階組件怎么使用”知識都有一定的了解,大家如果還想學習更多知識,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

慈利县| 富宁县| 石楼县| 岱山县| 忻城县| 名山县| 潢川县| 电白县| 贺州市| 莫力| 嘉峪关市| 鲁甸县| 临邑县| 辽阳县| 拉孜县| 大同县| 盐亭县| 乐亭县| 兴文县| 商洛市| 安平县| 商丘市| 仁怀市| 长武县| 通道| 涞水县| 集安市| 衡阳县| 潍坊市| 理塘县| 德保县| 山东省| 邵阳市| 通化市| 会宁县| 金秀| 会理县| 镇江市| 霸州市| 江门市| 龙泉市|