React進(jìn)階篇(一)高階組件

高階組件

高階組件(Higher Order Component,HOC)是React的一種設(shè)計(jì)模式记焊,用于增強(qiáng)現(xiàn)有組件的功能奶赔。

一個(gè)高階組件就是一個(gè)函數(shù)终吼,這個(gè)函數(shù)的輸入為組件烛愧,輸出為另一新組件。

根據(jù)輸入組件和輸出組件參數(shù)的關(guān)系希痴,可以分為:

  • 代理方式的高階組件
  • 繼承方式的高階組件
  • 函數(shù)式子組件
  • Mixin(不要使用)

1. 代理式高階組件(推薦)

例子如下:

// 為原組件WrappedComponent增加新的屬性newProps
function addNewProps(WrappedComponent, newProps) {
    return class WrappingComponent extends React.Component {
        render(){
            return <WrappedComponent {...this.props} {...newProps}>
        }
    }
}

特點(diǎn):

  1. 高階組件和被包裹組件有各自的生命周期捏悬;
  2. 可對(duì)原組件的props進(jìn)行增強(qiáng)或者刪減
  3. 渲染方式為 - return <WrappedComponent {...otherProps}/>

2.繼承式高階組件

繼承式高階組件常用于渲染劫持,例如润梯,當(dāng)用戶處于登陸狀態(tài)時(shí)过牙,允許組件渲染;否則渲染一個(gè)空組件纺铭。

function withAuth(WrappedComponent, newProps) {
    return class WrappingComponent extends WrappedComponent {
        render(){
            if (this.props.loggedIn) {
                this.props = {...this.props, ...newProps}
                return super.render();
            } else {
                return null;
            }
        }
    }
}

特點(diǎn):

  1. 只有一個(gè)生命周期
  2. 可以對(duì)原組件的props進(jìn)行增強(qiáng)或者刪減
  3. 渲染方式為 - return super.render();

3. 函數(shù)式子組件

前面兩種高階組件都會(huì)操作props寇钉,通過(guò)增減props而改變?cè)M件功能。

函數(shù)式子組件不會(huì)操作組件的props舶赔,但是扫倡,它的要求是:

  1. 父組件必須有子組件
  2. 子組件必須為函數(shù)

如下面的例子:

// 定義組件
class AddUserProp extends React.Component {
    render(){
        const user = 'mock user';
        return this.props.children(user);
    }
}

// 使用該組件
<AddUserProp>
{(user)=><div>user</div>}
</AddUserProp>

因?yàn)樽咏M件是函數(shù),所以這種模式非常靈活竟纳。

順著這個(gè)方式往下擴(kuò)展撵溃,我們可以發(fā)現(xiàn),這種父組件并沒有創(chuàng)建出新的組件锥累,而是將props屬性向注入到原組件內(nèi)缘挑。
這種設(shè)計(jì)模式就是“依賴注入”。當(dāng)A依賴B時(shí)桶略,并不要將A直接依賴B语淘,而是將B以接口的形式傳遞給A(通過(guò)函數(shù))诲宇。
所以,我們也可以讓父組件不包含子組件惶翻,直接將通過(guò)props函數(shù)來(lái)渲染組件姑蓝。

const Auth(props) {
   const user = getUser();
   if (user.login){
      const allProps = {...user, ...props};
      return (
        <React.Fragment>
          {props.login(allProps)}
        </React.Fragment> )
   } else {
       return (
        <React.Fragment>
          {props.nologin(props)}
        </React.Fragment> )
   }
}

// usage
<Auth
   login={props=><Login ...props/>}>
   nologin={props=><NoLogin ...props/>}
/>

4. MixIn

應(yīng)用場(chǎng)景:只能在React.createClass方式創(chuàng)建的組件類中使用,不能通過(guò)ES6 Class創(chuàng)建的組件中使用吕粗。

MixIn是一種反模式的設(shè)計(jì)纺荧,它可以繼承多個(gè)組件,包括其內(nèi)部狀態(tài)state颅筋。所以宙暇,很容易造成state混亂,官方不建議使用垃沦。

5. 注意事項(xiàng)

  • 不要在組件的render中使用高階組件客给。因?yàn)檎{(diào)用高階組件用押,每次都會(huì)返回一個(gè)新組件肢簿,所以,每次render蜻拨,前一次高階組件創(chuàng)建的組件都會(huì)被卸載池充,然后重新掛載,既影響效率缎讼,有丟失了組件及其子組件的狀態(tài)收夸。高階組件適合在組件外部使用
// 不好的應(yīng)用場(chǎng)景
render(){
    // 每次render血崭,enhance都會(huì)創(chuàng)建一個(gè)新組件卧惜,盡管被包裝組件沒有變化
    const EnhancedComponent = enhance(MyComponent);
    // 因?yàn)槭切陆M件,所以會(huì)經(jīng)歷舊組件的卸載和新組件的重新掛載
    return <EnhancedComponent />
}
  • 高階組件和父組件很類似夹纫。區(qū)別在于:高階組件是一個(gè)函數(shù)咽瓷,關(guān)注邏輯;父組件是一個(gè)組件舰讹,關(guān)注UI/DOM茅姜。如果邏輯和DOM不相關(guān)(如數(shù)據(jù)校驗(yàn),請(qǐng)求發(fā)送等)月匣,那么這部分邏輯適合放在高階組件里钻洒。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市锄开,隨后出現(xiàn)的幾起案子素标,更是在濱河造成了極大的恐慌,老刑警劉巖萍悴,帶你破解...
    沈念sama閱讀 206,723評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件糯钙,死亡現(xiàn)場(chǎng)離奇詭異粪狼,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)任岸,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,485評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門再榄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人享潜,你說(shuō)我怎么就攤上這事困鸥。” “怎么了剑按?”我有些...
    開封第一講書人閱讀 152,998評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵疾就,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我艺蝴,道長(zhǎng)猬腰,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,323評(píng)論 1 279
  • 正文 為了忘掉前任猜敢,我火速辦了婚禮姑荷,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘缩擂。我一直安慰自己鼠冕,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,355評(píng)論 5 374
  • 文/花漫 我一把揭開白布胯盯。 她就那樣靜靜地躺著懈费,像睡著了一般。 火紅的嫁衣襯著肌膚如雪博脑。 梳的紋絲不亂的頭發(fā)上憎乙,一...
    開封第一講書人閱讀 49,079評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音叉趣,去河邊找鬼泞边。 笑死,一個(gè)胖子當(dāng)著我的面吹牛君账,可吹牛的內(nèi)容都是我干的繁堡。 我是一名探鬼主播,決...
    沈念sama閱讀 38,389評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼乡数,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼椭蹄!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起净赴,我...
    開封第一講書人閱讀 37,019評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤绳矩,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后玖翅,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體翼馆,經(jīng)...
    沈念sama閱讀 43,519評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡割以,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,971評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了应媚。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片严沥。...
    茶點(diǎn)故事閱讀 38,100評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖中姜,靈堂內(nèi)的尸體忽然破棺而出消玄,到底是詐尸還是另有隱情,我是刑警寧澤丢胚,帶...
    沈念sama閱讀 33,738評(píng)論 4 324
  • 正文 年R本政府宣布翩瓜,位于F島的核電站,受9級(jí)特大地震影響携龟,放射性物質(zhì)發(fā)生泄漏兔跌。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,293評(píng)論 3 307
  • 文/蒙蒙 一峡蟋、第九天 我趴在偏房一處隱蔽的房頂上張望坟桅。 院中可真熱鬧,春花似錦层亿、人聲如沸桦卒。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,289評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至建蹄,卻和暖如春碌更,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背洞慎。 一陣腳步聲響...
    開封第一講書人閱讀 31,517評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工痛单, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人劲腿。 一個(gè)月前我還...
    沈念sama閱讀 45,547評(píng)論 2 354
  • 正文 我出身青樓旭绒,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親焦人。 傳聞我的和親對(duì)象是個(gè)殘疾皇子挥吵,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,834評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • 在目前的前端社區(qū),『推崇組合花椭,不推薦繼承(prefer composition than inheritance)...
    Wenliang閱讀 77,654評(píng)論 16 125
  • 其實(shí)之所以講到這里是因?yàn)?當(dāng)我們使用React的組件化開發(fā)Web應(yīng)用的時(shí)候,就會(huì)遇到這樣的問(wèn)題,很多組件需要某個(gè)功...
    井潤(rùn)閱讀 393評(píng)論 0 0
  • 作為一個(gè)合格的開發(fā)者忽匈,不要只滿足于編寫了可以運(yùn)行的代碼。而要了解代碼背后的工作原理矿辽;不要只滿足于自己的程序...
    六個(gè)周閱讀 8,422評(píng)論 1 33
  • 高階組件是對(duì)既有組件進(jìn)行包裝丹允,以增強(qiáng)既有組件的功能郭厌。其核心實(shí)現(xiàn)是一個(gè)無(wú)狀態(tài)組件(函數(shù)),接收另一個(gè)組件作為參數(shù)雕蔽,然...
    柏丘君閱讀 3,053評(píng)論 0 6
  • 事件系統(tǒng) 合成事件的綁定方式 Test 合成事件的實(shí)現(xiàn)機(jī)制:事件委派和自動(dòng)綁定折柠。 React合成事件系統(tǒng)的委托機(jī)制...
    cheneyg916閱讀 380評(píng)論 0 1