react中常見hook的使用方式與區別

一、什麼是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和繪製到屏幕之間。
useLayoutEffect與useEffect的區別.png

四、如何自定義hook? hook只能定義在函數式組件中,不能在普通函數中使用,若是咱們想要使用到上面的hook來封裝一些方法供不少個組件調用,這時候就須要自定義hook,自定義hook的命名就是在函數名前加 use,函數名由 saveInfo 改成 useSaveInfo 便可。

相關文章
相關標籤/搜索