React自定義Hook

1.什么是自定義 Hook?

通過(guò)自定義 Hook,可以對(duì)其它Hook的代碼進(jìn)行復(fù)用
官方文檔地址: https://react.docschina.org/docs/hooks-custom.html

注意點(diǎn): 在React中只有兩個(gè)地方可以使用Hook

  • 函數(shù)式組件中
  • 自定義Hook中
    如何自定義一個(gè)Hooks
    只要在函數(shù)名稱(chēng)前面加上use, 那么就表示這個(gè)函數(shù)是一個(gè)自定義Hook, 就表示可以在這個(gè)函數(shù)中使用其它的Hook
import React, {useEffect, useState} from 'react';
function Home() {
    useEffect(()=>{
        console.log('Home - 組件被掛載或者更新完成 -- 添加監(jiān)聽(tīng)');
        return ()=>{
            console.log('Home - 組件即將被卸載 -- 移出監(jiān)聽(tīng)');
        }
    });
    return (
        <div>Home</div>
    )
}
function About() {
    useEffect(()=>{
        console.log('About - 組件被掛載或者更新完成 -- 添加監(jiān)聽(tīng)');
        return ()=>{
            console.log('About - 組件即將被卸載 -- 移出監(jiān)聽(tīng)');
        }
    });
    return (
        <div>About</div>
    )
}
function App() {
    const [show, setShow] = useState(true);
    return (
        <div>
            {show && <Home/>}
            {show && <About/>}
            <button onClick={()=>{setShow(!show)}}>切換</button>
        </div>
    )
}
export default App;
點(diǎn)擊按鈕.png
清空控制臺(tái)再次點(diǎn)擊按鈕.png

但是不難發(fā)現(xiàn),其實(shí)兩個(gè)組件的useEffect里頭的方法高度相似,容易造成代碼冗余巍膘,這個(gè)時(shí)候我們可以將方法抽取出來(lái),嘗試一下吧!

import React, {useEffect, useState} from 'react';
import './app.css'
function addListenr(name) {
    useEffect(()=>{
        console.log(name, ' - 組件被掛載或者更新完成 -- 添加監(jiān)聽(tīng)');
        return ()=>{
            console.log(name, ' - 組件即將被卸載 -- 移出監(jiān)聽(tīng)');
        }
    });
}
function Home() {
    addListenr('Home');
    return (
        <div>Home</div>
    )
}
function About() {
addListenr('About');
    return (
        <div>About</div>
    )
}
function App() {
    const [show, setShow] = useState(true);
    return (
        <div>
            {show && <Home/>}
            {show && <About/>}
            <button onClick={()=>{setShow(!show)}}>切換</button>
        </div>
    )
}
export default App;

但是去控制發(fā)現(xiàn)會(huì)報(bào)錯(cuò)


報(bào)錯(cuò).png

但是為什么會(huì)報(bào)錯(cuò)呢唆香?
原因是useEffect屬于react-hooks,他只能在函數(shù)式組件內(nèi)使用吨艇,現(xiàn)在把它定義在外面自然會(huì)報(bào)錯(cuò)
怎么解決呢躬它?使用自定義的Hooks,通過(guò)函數(shù)命名來(lái)定義,只需將addlistener函數(shù)名改為useAddListener即可

function useAddListenr(name) {
    useEffect(()=>{
        console.log(name, ' - 組件被掛載或者更新完成 -- 添加監(jiān)聽(tīng)');
        return ()=>{
            console.log(name, ' - 組件即將被卸載 -- 移出監(jiān)聽(tīng)');
        }
    });
}

完整代碼[final solution]:

import React, {useEffect, useState} from 'react';
function useAddListenr(name) {
    useEffect(()=>{
        console.log(name, ' - 組件被掛載或者更新完成 -- 添加監(jiān)聽(tīng)');
        return ()=>{
            console.log(name, ' - 組件即將被卸載 -- 移出監(jiān)聽(tīng)');
        }
    });
}
function Home() {
    useAddListenr('Home');
    return (
        <div>Home</div>
    )
}
function About() {
    useAddListenr('About');
    return (
        <div>About</div>
    )
}
function App() {
    const [show, setShow] = useState(true);
    return (
        <div>
            {show && <Home/>}
            {show && <About/>}
            <button onClick={()=>{setShow(!show)}}>切換</button>
        </div>
    )
}
export default App;

注意點(diǎn)二:在企業(yè)開(kāi)發(fā)中, 但凡需要抽取代碼, 但凡被抽取的代碼中用到了其它的Hook, 那么就必須把這些代碼抽取到自定義Hook中

import React, {createContext, useContext} from 'react';
const UserContext = createContext({});
const InfoContext = createContext({});
function useGetContext() {
//用到了useContext的hook东涡,因此必須使用自定義hook來(lái)抽取代碼進(jìn)行優(yōu)化
    const user = useContext(UserContext);
    const info = useContext(InfoContext);
    return [user, info]
}
function Home() {
    // const user = useContext(UserContext);
    // const info = useContext(InfoContext);
    const [user, info] = useGetContext();
    return (
        <div>
            <p>{user.name}</p>
            <p>{user.age}</p>
            <p>{info.gender}</p>
            <hr/>
        </div>
    )
}
function About() {
    // const user = useContext(UserContext);
    // const info = useContext(InfoContext);
    const [user, info] = useGetContext();
    return (
        <div>
            <p>{user.name}</p>
            <p>{user.age}</p>
            <p>{info.gender}</p>
            <hr/>
        </div>
    )
}
function App() {
    return (
        <UserContext.Provider value={{name:'yiya_xiaoshan', age:18}}>
            <InfoContext.Provider value={{gender:'female'}}>
                <Home/>
                <About/>
            </InfoContext.Provider>
        </UserContext.Provider>
    )
}
export default App;

不懼怕困難冯吓,走出舒適區(qū)真正的成長(zhǎng)總是有汗水和艱辛造就的,安逸只會(huì)讓我們慢慢失去活力
不知不覺(jué)已經(jīng)到了學(xué)習(xí)react的尾聲疮跑,小單想為自己鼓掌组贺,加油!堅(jiān)持~


小單真棒.gif
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末祖娘,一起剝皮案震驚了整個(gè)濱河市失尖,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌渐苏,老刑警劉巖雹仿,帶你破解...
    沈念sama閱讀 217,826評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異整以,居然都是意外死亡胧辽,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,968評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)公黑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)邑商,“玉大人,你說(shuō)我怎么就攤上這事凡蚜∪硕希” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,234評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵朝蜘,是天一觀的道長(zhǎng)恶迈。 經(jīng)常有香客問(wèn)我,道長(zhǎng)谱醇,這世上最難降的妖魔是什么暇仲? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,562評(píng)論 1 293
  • 正文 為了忘掉前任步做,我火速辦了婚禮,結(jié)果婚禮上奈附,老公的妹妹穿的比我還像新娘全度。我一直安慰自己,他們只是感情好斥滤,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,611評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布将鸵。 她就那樣靜靜地躺著,像睡著了一般佑颇。 火紅的嫁衣襯著肌膚如雪顶掉。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,482評(píng)論 1 302
  • 那天挑胸,我揣著相機(jī)與錄音痒筒,去河邊找鬼。 笑死嗜暴,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的议蟆。 我是一名探鬼主播闷沥,決...
    沈念sama閱讀 40,271評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼咐容!你這毒婦竟也來(lái)了舆逃?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,166評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤戳粒,失蹤者是張志新(化名)和其女友劉穎路狮,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體蔚约,經(jīng)...
    沈念sama閱讀 45,608評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡奄妨,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,814評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了苹祟。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片砸抛。...
    茶點(diǎn)故事閱讀 39,926評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖树枫,靈堂內(nèi)的尸體忽然破棺而出直焙,到底是詐尸還是另有隱情,我是刑警寧澤砂轻,帶...
    沈念sama閱讀 35,644評(píng)論 5 346
  • 正文 年R本政府宣布奔誓,位于F島的核電站,受9級(jí)特大地震影響搔涝,放射性物質(zhì)發(fā)生泄漏厨喂。R本人自食惡果不足惜和措,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,249評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望杯聚。 院中可真熱鬧臼婆,春花似錦、人聲如沸幌绍。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,866評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)傀广。三九已至颁独,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間伪冰,已是汗流浹背誓酒。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,991評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留贮聂,地道東北人靠柑。 一個(gè)月前我還...
    沈念sama閱讀 48,063評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像吓懈,于是被迫代替她去往敵國(guó)和親歼冰。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,871評(píng)論 2 354

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

  • 本文首發(fā)于公眾號(hào)【一個(gè)老碼農(nóng)】 什么是hook Hook是 React 16.8 的新增特性耻警。它通常與函數(shù)式組件同...
    劍老師閱讀 542評(píng)論 0 0
  • 在react類(lèi)組件中隔嫡,有組件間可共享邏輯時(shí),一般用高階組件的方式對(duì)公共邏輯進(jìn)行復(fù)用甘穿,在react函數(shù)組件中腮恩,hoo...
    湯姆威廉斯閱讀 1,599評(píng)論 0 0
  • 什么是自定義hook? 使用自定義hook可以將某些組件邏輯提取到可重用的函數(shù)中温兼。 自定義hook是一個(gè)從use開(kāi)...
    魂斗驢閱讀 6,455評(píng)論 1 3
  • 自定義 Hook 必須以 “use” 開(kāi)頭嗎秸滴?**必須如此。這個(gè)約定非常重要募判。不遵循的話缸榛,由于無(wú)法判斷某個(gè)函數(shù)是否...
    怒默閱讀 200評(píng)論 0 1
  • 前言 哈嘍内颗,大家好,我是海怪敦腔。 最近把項(xiàng)目里的 utils 以及 components 里的東西都測(cè)完了均澳,算是完成...
    寫(xiě)代碼的海怪閱讀 336評(píng)論 0 0