日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪(fǎng)問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

React 高阶组件HOC详解

發(fā)布時(shí)間:2023/12/14 编程问答 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 React 高阶组件HOC详解 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

參考鏈接:

https://juejin.cn/post/6844903815762673671

https://juejin.cn/post/6844904050236850184

前言

高階組件與自定義hooks是React 目前流行的狀態(tài)邏輯復(fù)用的兩種解決方案

1.高階組件是什么

高階組件就是一個(gè)函數(shù),且該函數(shù)接受一個(gè)組件作為參數(shù),并返回一個(gè)新的組件。

高階組件(HOC)是React中的高級(jí)技術(shù),用來(lái)重用組件邏輯。但高階組件本身并不是React API。它只是一種模式,這種模式是由React自身的組合性質(zhì)必然產(chǎn)生的。

HOC簡(jiǎn)單例子:

//HOC function visible(WrappedComponent) {return class extends Component {render() {const { visible, ...props } = this.props;if (visible === false) return null;return <WrappedComponent {...props} />;}} }// 用HOC包裹組件 class Example extends Component {render() {return <span>示例組件</span>;} } export default HOC(Example) //或者用decorator方式 @HOC class Example extends Component {render() {return <span>示例組件</span>;} } export default Example//使用 <Example visible={false}/>

上面的代碼就是一個(gè)HOC的簡(jiǎn)單應(yīng)用,函數(shù)接收一個(gè)組件作為參數(shù),并返回一個(gè)新組件,新組建可以接收一個(gè)visible props,根據(jù)visible的值來(lái)判斷是否渲染傳入的組件。

2 高階組件實(shí)現(xiàn)方式

2.1屬性代理

將一個(gè)React組件作為參數(shù)傳入函數(shù)中,函數(shù)返回一個(gè)自定義的組件。該自定義組件的render函數(shù)中返回傳入的React組件。

由此可以代理并操作傳入的React組件的props,并且決定如何渲染,實(shí)際上 ,這種方式生成的高階組件就是原組件的父組件,上面的函數(shù)visible就是一個(gè)HOC屬性代理的實(shí)現(xiàn)方式。

這種實(shí)現(xiàn)方式下,HOC容器組件和傳入組件的生命周期調(diào)用順序和父,子組件的生命周期順序是一致的。類(lèi)似堆棧調(diào)用(先入后出)

function proxyHOC(WrappedComponent) {return class extends Component {render() {return <WrappedComponent {...this.props} />;}} } //使用示例 class Example extends Component {render() {return <input name="name" {...this.props.name} />;} } export default HOC(Example)

通過(guò)屬性代理實(shí)現(xiàn)的HOC可具有以下功能:

(1)操作props

可以對(duì)傳入組件的props進(jìn)行增加、修改、刪除或者根據(jù)特定的props進(jìn)行特殊的操作。

注意,使用HOC包裹后的組件,在給組件傳入props時(shí)實(shí)際傳入到了HOC的container容器組件中,如不需要操作props,請(qǐng)務(wù)必在容器組件中將props再度傳給傳入組件,否則傳入組件不會(huì)接收到props

function proxyHOC(WrappedComponent) {return class Container extends Component {render() {const newProps = {...this.props,user: "ConardLi"}return <WrappedComponent {...newProps} />;}} }

(2)獲取refs引用

高階組件中可獲取傳入組件的ref,通過(guò)ref獲取組件的實(shí)例(即拿到傳入組件實(shí)例的this),如下面的代碼,當(dāng)程序初始化完成后調(diào)用原組件的log方法。

function refHOC(WrappedComponent) {return class Container extends Component {componentDidMount() {this.wapperRef.log()}render() {return <WrappedComponent {...this.props} ref={ref => { this.wapperRef = ref }} />;}} }

注意:HOC包裹的組件默認(rèn)無(wú)法在外部調(diào)用時(shí)拿到原組件refs引用

雖然高階組件的約定是將所有 props 傳遞給被包裝組件,但這對(duì)于 refs 并不適用。那是因?yàn)?ref 實(shí)際上并不是一個(gè) prop。就像 key 一樣,它是由 React 專(zhuān)門(mén)處理的。如果將 ref 添加到 HOC 的返回組件中,則 ref 引用指向容器組件,而不是被包裝組件。

這個(gè)問(wèn)題的解決方案是通過(guò)使用 React.forwardRef API(React 16.3 中引入)。

(3)抽象state

// 高階組件 function HOC(WrappedComponent) {return class Container 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} />;}}; }// 使用 class Example extends Component {render() {return <input name="name" {...this.props.name} />;} } export default HOC(Example) //或者 @HOC class Example extends Component {render() {return <input name="name" {...this.props.name} />;} }

在這個(gè)例子中,我們把 input 組件中對(duì) name這個(gè)prop在高階組件中進(jìn)行了重定義的覆蓋(用value和onChange 代替),這就有效地抽象了同樣的 state 操作。使得input組件由非受控組件變成了受控組件

(4)操作組件的static方法

可以對(duì)傳入組件的static靜態(tài)方法進(jìn)行獲取調(diào)用,增加、修改、刪除

function refHOC(WrappedComponent) {return class Container extends Component {componentDidMount() {//獲取static方法console.log(WrappedComponent.staticMethod)}//新增static方法WrappedComponent.addMethod1=()=>{}render() {return <WrappedComponent {...this.props} ref={ref => { this.wapperRef = ref }} />;}} }

但當(dāng)你將 HOC 應(yīng)用于組件時(shí),原始組件將使用容器組件進(jìn)行包裝。這意味著容器組件默認(rèn)沒(méi)有傳入組件的任何靜態(tài)方法,即無(wú)法在其他地方引入組件時(shí)拿到其靜態(tài)方法。所以與props同理,請(qǐng)務(wù)必將傳入組件的靜態(tài)方法拷貝到容器組件上

(5)根據(jù)props實(shí)現(xiàn)條件渲染

根據(jù)特定的props決定傳入組件是否渲染(如最上面的基本HOC例子)

function visibleHOC(WrappedComponent) {return class extends Component {render() {if (this.props.visible === false) return null;return <WrappedComponent {...props} />;}} }

(6)用其他元素包裹傳入的組件

在HOC的容器組件中將原組件通過(guò)其他元素再包裹起來(lái),從而實(shí)現(xiàn)布局或者修改樣式的目的:

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

2.2 反向繼承

返回一個(gè)組件,該組件繼承傳入組件,在render中調(diào)用原組件的render。

由于繼承了原組件,能通過(guò)this訪(fǎng)問(wèn)到原組件的生命周期、props、state、render等,相比屬性代理它能操作更多的屬性。

這種實(shí)現(xiàn)方式下,HOC組件和傳入組件的生命周期調(diào)用順序與隊(duì)列類(lèi)似(先進(jìn)先出)

function inheritHOC(WrappedComponent) {return class extends WrappedComponent {render() {return super.render();}} }

通過(guò)反向繼承實(shí)現(xiàn)的HOC,相比屬性代理具有以下額外的功能:

(1)渲染劫持

渲染劫持指的就是高階組件可以控制 WrappedComponent 的渲染過(guò)程,并渲染各種各樣的結(jié)果。我們可以在這個(gè)過(guò)程中在任何 React 元素輸出的結(jié)果中讀取、增加、修改、刪除 props,或讀取或修改 React 元素樹(shù),或條件顯示元素樹(shù),又或是用樣式控制包裹元素樹(shù)。

上面屬性代理提到的條件渲染,其實(shí)也是渲染劫持的一種實(shí)現(xiàn)。

如果元素樹(shù)中包括了函數(shù)類(lèi)型的 React 組件,就不能操作組件的子組件

渲染劫持實(shí)示例:

function hijackHOC(WrappedComponent) {return class extends WrappedComponent {render() {const tree = super.render();let newProps = {};if (tree && tree.type === "input") {newProps = { value: "渲染被劫持了" };}const props = Object.assign({}, tree.props, newProps);const newTree = React.cloneElement(tree, props, tree.props.children);return newTree;}} }

(2)劫持傳入組件生命周期

因?yàn)榉聪蚶^承方式實(shí)現(xiàn)的高階組件返回的新組件是繼承于傳入組件,所以當(dāng)新組件定義了同樣的方法時(shí),將會(huì)會(huì)覆蓋父類(lèi)(傳入組件)的實(shí)例方法,如下面代碼所示:

function HOC(WrappedComponent){// 繼承了傳入組件return class HOC extends WrappedComponent {// 注意:這里將重寫(xiě) componentDidMount 方法componentDidMount(){...}render(){//使用 super 調(diào)用傳入組件的 render 方法return super.render();}} }

(3)操作傳入組件state

反向繼承方式實(shí)現(xiàn)的高階組件中可以讀取、編輯和刪除傳入組件實(shí)例中的 state,如下面代碼所示:

function debugHOC(WrappedComponent) {return class extends WrappedComponent {render() {console.log("props", this.props);console.log("state", this.state);return (<div className="debuging">{super.render()}</div>)}} }

操作傳入組件的state可能會(huì)讓 WrappedComponent 組件內(nèi)部狀態(tài)變得一團(tuán)糟。大部分的高階組件都應(yīng)該限制讀取或增加 state,尤其是后者,可以通過(guò)重新命名 state,以防止混淆。

2.3 兩種方式對(duì)比

  • 屬性代理是從“組合”的角度出發(fā),這樣有利于從外部去操作 WrappedComponent,可以操作的對(duì)象是 props,或者在 WrappedComponent 外面加一些攔截器,控制器等。
  • 反向繼承則是從“繼承”的角度出發(fā),是從內(nèi)部去操作 WrappedComponent,也就是可以操作組件內(nèi)部的 state ,生命周期,render函數(shù)等等。

3. 高階組件實(shí)際應(yīng)用

(1)邏輯復(fù)用

多個(gè)頁(yè)面組件存在代碼結(jié)構(gòu)和需求相似的情況,只是一些傳參和數(shù)據(jù)不同,存在較多重復(fù)性代碼。使用高階組件進(jìn)行統(tǒng)一包裹封裝即可

下面是兩個(gè)結(jié)構(gòu)和需求相似的頁(yè)面組件:

// views/PageA.js import React from "react"; import fetchMovieListByType from "../lib/utils"; import MovieList from "../components/MovieList";class PageA extends React.Component {state = {movieList: [],}/* ... */async componentDidMount() {const movieList = await fetchMovieListByType("comedy");this.setState({movieList,});}render() {return <MovieList data={this.state.movieList} emptyTips="暫無(wú)喜劇"/>} } export default PageA; // views/PageB.js import React from "react"; import fetchMovieListByType from "../lib/utils"; import MovieList from "../components/MovieList";class PageB extends React.Component {state = {movieList: [],}// ...async componentDidMount() {const movieList = await fetchMovieListByType("action");this.setState({movieList,});}render() {return <MovieList data={this.state.movieList} emptyTips="暫無(wú)動(dòng)作片"/>} } export default PageB;

將重復(fù)邏輯抽離成一個(gè)HOC:

// HOC import React from "react"; const withFetchingHOC = (WrappedComponent, fetchingMethod, defaultProps) => {return class extends React.Component {async componentDidMount() {const data = await fetchingMethod();this.setState({data,});}render() {return (<WrappedComponent data={this.state.data} {...defaultProps} {...this.props} />);}} }

使用示例:

// 使用: // views/PageA.js import React from "react"; import withFetchingHOC from "../hoc/withFetchingHOC"; import fetchMovieListByType from "../lib/utils"; import MovieList from "../components/MovieList"; const defaultProps = {emptyTips: "暫無(wú)喜劇"}export default withFetchingHOC(MovieList, fetchMovieListByType("comedy"), defaultProps);// views/PageB.js import React from "react"; import withFetchingHOC from "../hoc/withFetchingHOC"; import fetchMovieListByType from "../lib/utils"; import MovieList from "../components/MovieList"; const defaultProps = {emptyTips: "暫無(wú)動(dòng)作片"}export default withFetchingHOC(MovieList, fetchMovieListByType("action"), defaultProps);;// views/PageOthers.js import React from "react"; import withFetchingHOC from "../hoc/withFetchingHOC"; import fetchMovieListByType from "../lib/utils"; import MovieList from "../components/MovieList"; const defaultProps = {...}export default withFetchingHOC(MovieList, fetchMovieListByType("some-other-type"), defaultProps);

上面設(shè)計(jì)的高階組件 withFetchingHOC,把不一樣的部分(組件和獲取數(shù)據(jù)的方法) 抽離到外部作為傳入,從而實(shí)現(xiàn)頁(yè)面的復(fù)用。

(2)權(quán)限控制

function auth(WrappedComponent) {return class extends Component {render() {const { visible, auth, display = null, ...props } = this.props;if (visible === false || (auth && authList.indexOf(auth) === -1)) {return display}return <WrappedComponent {...props} />;}} }

authList是我們?cè)谶M(jìn)入程序時(shí)向后端請(qǐng)求的所有權(quán)限列表,當(dāng)組件所需要的權(quán)限不在傳入權(quán)限列表中,或者設(shè)置的 visible是false,我們將其顯示為傳入的組件樣式,或者null。我們可以將任何需要進(jìn)行權(quán)限校驗(yàn)的組件應(yīng)用HOC:

@authclass Input extends Component { ... }@authclass Button extends Component { ... }<Button auth="user/addUser">添加用戶(hù)</Button><Input auth="user/search" visible={false} >添加用戶(hù)</Input>

4. 其他技巧:

(1)高階組件參數(shù)

有時(shí),我們調(diào)用高階組件時(shí)需要傳入一些參數(shù),這可以用非常簡(jiǎn)單的方式來(lái)實(shí)現(xiàn):

import React, { Component } from "React"; function HOCFactoryFactory(...params) { // 可以做一些改變 params 的事return function HOCFactory(WrappedComponent) { return class HOC extends Component { render() { return <WrappedComponent {...this.props} />; } } } }

當(dāng)你使用的時(shí)候,可以這么寫(xiě):

HOCFactoryFactory(params)(WrappedComponent) // 或者 @HOCFatoryFactory(params) class WrappedComponent extends React.Component{}

(2)高階組件命名

當(dāng)包裹一個(gè)高階組件時(shí),我們失去了原始 WrappedComponent 的 displayName,而組件名字是方便我們開(kāi)發(fā)與調(diào)試的重要屬性

HOC.displayName = `HOC(${getDisplayName(WrappedComponent)})`; // 或者 class HOC extends ... { static displayName = `HOC(${getDisplayName(WrappedComponent)})`; ...

然后通過(guò)HOC.displayName來(lái)獲取即可

總結(jié)

以上是生活随笔為你收集整理的React 高阶组件HOC详解的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。