一、什麼是hook?
react hook是react 16.8推出的方法,可以讓函數式組件像類式組件同樣擁有state、ref、生命週期等屬性。react
二、爲何要出現hook?
函數式組件是全局當中一個普通函數,在非嚴格模式下this指向window,可是react內部開啓了嚴格模式,此時this指向undefined,沒法像類式組件同樣使用state、ref,函數式組件定義的變量都是局部的,當組件進行更新時會從新定義,也沒法存儲,因此在hook出現以前,函數式組件有很大的侷限性,一般狀況下都會使用類式組件來進行代碼的編寫。數組
三、有哪些經常使用的hook?
(1) useState
使函數式組件也能保存狀態的一個hook,這個hook的入參是狀態的初始值,返回值是一個數組,數組裏第一個參數爲狀態的值,第二個參數爲修改狀態的方法。性能優化
// 初始化 const [ count, setCount ] = useState(0) // 更新 setCount(count+1)
(2) useEffect
函數式組件用來模擬生命週期的hook,能夠模擬組件掛載完成、更新完成、即將卸載三個階段,即componentDidMount、componentDidUpdate、componentWillUnmount。dom
useEffect的一個參數爲函數,表示組件掛載、更新時執行的內容,在函數裏再返回一個函數,表示組件即將卸載時調用的函數。ide
第二個參數爲可選項,可傳入數組,數組裏能夠爲空,表示不依賴任何狀態的變化,即只在組件即將掛載時執行,後續任何狀態發生了變化,都不調用此hook。數組裏也能夠定義一或多個狀態,表示每次該狀態變化時,都會執行此hook。函數
useEffect(()=>{ // 這樣模擬的是 componentDidMount }, []) useEffect(()=>{ // 這樣模擬的是componentDidMount 以及當count發生變化時執行componentDidUpdate }, [count]) useEffect(()=>{ return ()=>{ // 這樣模擬的是 componentWillUnmount } }, [])
(3) useContext
在沒有hook以前,咱們一般都會經過 xxxContext.Provider 和 xxxContext.Consumer 的方式來傳遞和獲取context的值,使用hook以後,傳遞context的方式不變,但子元素獲取context的方式變得更加的簡潔。性能
// 之前的定義方式 const CountContext = React.createContext() <CountContext.Provider value={{ count: 10 }}> <...自定義的組件> </CountContext.Provider> // 子元素 <CountContext.Consumer> { value => { console.log(value.count) }} //10 </CountContext.Consumer> //使用context的獲取方式 const countObj = useContext(CountContext) console.log(countObj.count) // 10
(4) useRef
useRef和類式組件中createRef用法比較相似,返回一個ref對象,這個對象在函數的整個生命週期都不變,根據這個特性,有兩種比較常見的用法。優化
① 用於dom元素或者組件上,經過current屬性能夠獲取到dom元素或者類式組件的實例對象。須要注意的是,不管是useRef仍是createRef或者是回調形式、字符串形式的ref,都是不能直接給函數式組件定義的,由於函數式組件的this指向undefined,沒有實例對象,只能經過forwardRef定義到函數式組件中的某個dom元素。this
// 這樣就將傳遞給函數式組件的ref綁定在了函數式組件內部的input標籤上 import React, { useRef, forwardRef } from 'react' // 使用函數表達式的方式定義了一個函數式組件 const InputCom = forwardRef((props, ref) => { return <input ref={ref}/> }) export default function refDemo(){ const comRef = useRef() return(<div> <InputCom ref={comRef}/> </div>) }
② 保存一個數據,該數據若是不手動修改,它在整個生命週期中都不變spa
const [ count, setCount ] = useState(0) const prevCount = useState(count) // 當count發生變化時,組件更新,對count的前一次數據進行保存 useEffect(()=>{ prevCount.current = count }, [count])
(5) useReducer
useReducer至關因而useState的升級版,做用與useState相似,都是用來保存狀態,但它的不一樣點在於能夠定義一個reducer的純函數,來處理複雜數據。
// 定義一個處理數據的reducer純函數 function reducer(prevState, action){ switch(action.type){ case 'increment': return {...prevState, count: prevState.count + 1 } case 'decrement': return {...prevState, count: prevState.count - 1 } default: return prevState } } // 初始化狀態 const [ count, dispatch ] = useReducer(reducer, { count: 0 }) // 修改狀態,此時的修改須要派發一個action,讓傳入的reducer函數進行處理 dispatch({ type: 'increment' })
(6) useCallback
函數式組件中,每一次更新狀態,自定義的函數都要進行從新的聲明和定義,若是函數做爲props傳遞給子組件,會形成子組件沒必要要的從新渲染,有時候子組件並無使用到父組件發生變化的狀態,此時可使用useCallback來進行性能優化,它會爲函數返回一個記憶的值,若是依賴的狀態沒有發生變化,那麼則不會從新建立該函數,也就不會形成子組件沒必要要的從新渲染。
import React, { useState, useCallback, memo } from 'react' const AddBtn = memo((props)=>{ // 使用函數表達式的方式定義了一個函數式組件 return<button onClick={props.increment}>+1</button> }) export default function CallBackPerformance(){ const [ count, setCount ] = useState(0) const [ show, setShow ] = useState(true) const increment1 = () => { console.log('increment1被調用了') setCount(count+1) } const increment2 = useCallback(()=>{ // 使用了useCallback來優化的函數 console.log('increment2被調用了') setCount(count+1) },[count]) return(<div> <div>當前計數:{count}</div> <AddBtn increment={increment1} name="1"/> <AddBtn increment={increment2} name="2"/> <button onClick={e => setShow(!show)}>切換show</button> </div>) } // 當show這個狀態發生變化時,子組件increment1會從新渲染,increment2不會從新渲染
(7) useMemo
useMemo也是返回一個記憶的值,若是依賴的內容沒有發生改變的話,這個值也不會發生變化,useMemo與useCallback的不一樣點在於useMemo須要在傳入的函數裏須要return 一個值,這個值能夠是對象、函數,格式以下。
useMemo(()=>{ return { count } }, [count]) // 使用useCallback時 const increment2 = useCallback(()=>{ setCount(count+1) },[count]) // 使用useMemo模擬useCallback const increment2 = useCallback(()=>{ return ()=>{ setCount(count+1) } },[count]) // useMemo的應用場景,當要進行一些複雜的計算時, //計算的值沒有發生變化,並不須要每一次更新都從新計算 import React, { useState, useMemo } from 'react' const calculateNum = (count) => { console.log('total從新計算了') let total = 0 for(let i = 0; i <= count; i++){ total += i } return total } export default function ComplexUseMemo(){ const [ count, setCount ] = useState(10) const [ show, setShow ] = useState(true) const total = useMemo(()=>{ return calculateNum(count) }, [count]) return(<div> <div>{total}</div> <button onClick={e=>setCount(count+1)}>+1</button> <button onClick={e=>setShow(!show)}>切換show</button> </div>) }
(8) useImperativeHandle
這個是與forwardRef配合來使用的,當咱們對函數式組件使用forwardRef將ref指定了dom元素以後,那就父組件就能夠任意的操做指定的dom元素,使用useImperativeHandle就是爲了控制這樣的一種行爲,指定父元素可操做的子元素的方法。
import React, { useRef, useImperativeHandle, forwardRef } from 'react' const InputComp = forwardRef((props, ref)=>{ const childInputRef = useRef() useImperativeHandle(ref, ()=>({ focus: ()=>{ childInputRef.current.focus() } }), [childInputRef.current]) return<input ref={childInputRef}></input> }) export default function ImperativeHookDemo() { const inputRef = useRef() return(<div> <InputComp ref={inputRef}/> <button onClick={e=>inputRef.current.focus()}>聚焦</button> </div>) }
(9) useLayoutEffect
這個方法與useEffect相似,只是執行的順序稍有不一樣,useEffect是在組件渲染繪製到屏幕上以後,useLayoutEffect是render和繪製到屏幕之間。
四、如何自定義hook? hook只能定義在函數式組件中,不能在普通函數中使用,若是咱們想要使用到上面的hook來封裝一些方法供不少個組件調用,這時候就須要自定義hook,自定義hook的命名就是在函數名前加 use,函數名由 saveInfo 改成 useSaveInfo 便可。