溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

react中useEffect閉包問(wèn)題怎么解決

發(fā)布時(shí)間:2022-04-20 10:04:19 來(lái)源:億速云 閱讀:825 作者:iii 欄目:大數(shù)據(jù)

這篇文章主要介紹“react中useEffect閉包問(wèn)題怎么解決”的相關(guān)知識(shí),小編通過(guò)實(shí)際案例向大家展示操作過(guò)程,操作方法簡(jiǎn)單快捷,實(shí)用性強(qiáng),希望這篇“react中useEffect閉包問(wèn)題怎么解決”文章能幫助大家解決問(wèn)題。

問(wèn)題代碼

看一段因?yàn)閡seEffect導(dǎo)致的閉包問(wèn)題代碼

const btn = useRef();
const [v, setV] = useState('');

useEffect(() => {
    let clickHandle = () => {
        console.log('v:', v);
    }
    btn.current.addEventListener('click', clickHandle)
    
    return () => {
        btn.removeEventListener('click', clickHandle)
    }
}, []);
    
const inputHandle = e => {
    setV(e.target.value)
}

return (
        <>
            <input value={v} onChange={inputHandle} />
            <button ref={btn} >測(cè)試</button>
        </>
    )

useEffect的依賴項(xiàng)數(shù)組為空,所以在頁(yè)面渲染完成之后,內(nèi)部代碼只會(huì)執(zhí)行一次,頁(yè)面銷毀再執(zhí)行一次。此時(shí)在輸入框中輸入任意字符,再點(diǎn)擊測(cè)試按鈕,得到的輸出為空,之后無(wú)論如何輸入任何字符,再點(diǎn)擊測(cè)試按鈕時(shí),輸出的結(jié)果仍為空。

為什么會(huì)這樣呢?其實(shí)就是閉包所造成的。

產(chǎn)生原因

函數(shù)的作用域在函數(shù)定義的時(shí)候就決定了

給btn注冊(cè)點(diǎn)擊事件時(shí),作用域如下:

react中useEffect閉包問(wèn)題怎么解決

能訪問(wèn)到的自由變量v此時(shí)還是空值。當(dāng)點(diǎn)擊事件觸發(fā)時(shí),執(zhí)行點(diǎn)擊回調(diào)函數(shù),此時(shí)先創(chuàng)建執(zhí)行上下文,會(huì)拷貝作用域鏈到執(zhí)行上下文中。

  • 如果未在輸入框內(nèi)輸入字符,此時(shí)點(diǎn)擊拿到的v還是原來(lái)那個(gè)v

  • 如果在輸入框內(nèi)輸入了字符,此時(shí)調(diào)用了setV修改了state,頁(yè)面觸發(fā)render,組件內(nèi)部代碼會(huì)重新執(zhí)行一遍,重新聲明了一個(gè)v,v就不再是原來(lái)那個(gè)v,這里點(diǎn)擊事件里作用域中的v還是舊的v,這是兩個(gè)不同的v

產(chǎn)生場(chǎng)景

  • 事件綁定。比如示例代碼中,在頁(yè)面最初渲染完成后只綁定一次事件的情況,比如使用echarts,在useEffect中獲取echarts的實(shí)例并綁定事件

  • 定時(shí)器。頁(yè)面加載后注冊(cè)一個(gè)定時(shí)器,定時(shí)器內(nèi)的函數(shù)也會(huì)產(chǎn)生如此的閉包問(wèn)題。

解決辦法

針對(duì)這個(gè)閉包問(wèn)題下面大致給出5種解決辦法

1. 以賦值方式直接修改v,并將修改v的方法用useCallback包裹起來(lái)

將修改v的方法用useCallback包裹起來(lái),被useCallback包裹的函數(shù)將被緩存,由于依賴項(xiàng)的數(shù)組為空,所以這里直接賦值的方式修改的v是舊的v,此種方法不推薦,因?yàn)閟etState才是官方推薦的修改state的方式,這里仍然使用setV只是為了觸發(fā)rerender

// v 的聲明 由 const 改為 var,方便直接修改
var [v, setV] = useState('');

const inputHandle = useCallback(e => {
    let { value } = e.target
    v = value
    setV(value)
}, [])

2. 給useEffect的依賴項(xiàng)加上v

這也許是大多數(shù)人首先想到的辦法,既然v是舊的,那么每次v更新的時(shí)候,重新注冊(cè)一次事件不就行了,但是這樣的會(huì)導(dǎo)致每次v更新都得重新注冊(cè),理論應(yīng)該只需要注冊(cè)一次的事件變成了多次。

3. 避免v被重新聲明

以let或var的方式聲明某個(gè)變量代替v,直接修改這個(gè)變量,而不是要setState相關(guān)函數(shù)觸發(fā)render,這樣就不會(huì)被重新聲明,點(diǎn)擊的回調(diào)函數(shù)里就能拿到“最新”的值,但這個(gè)方法更不推薦,就此示例來(lái)說(shuō),input組件由于沒(méi)有rerender而至始至終都是顯示空值,不符合操作預(yù)期。

4. 使用useRef代替useState

const btn = useRef();
const vRef = useRef('');
const [v, setV] = useStat('');

useEffect(() => {
    let clickHandle = () => {
        console.log('v:', vRef.current);
    }
    btn.current.addEventListener('click', clickHandle)
    
    return () => {
        btn.removeEventListener('click', clickHandle)
    }
}, []);

const inputHandle = e => {
    let { value } = e.target
    vRef.current = value
    setV(value)
}

return (
        <>
            <input value={v} onChange={inputHandle} />
            <button ref={btn} >測(cè)試</button>
        </>
    )

useRef的方案之所以有效,是因?yàn)槊看蝘nput的change修改的是vRef這個(gè)對(duì)象的current屬性,而vRef始終是那個(gè)vRef,即使rerender,由于vRef是對(duì)象,所以變量存儲(chǔ)在棧內(nèi)存中的值是該對(duì)象在堆內(nèi)存中的地址,只是一個(gè)引用,只修改對(duì)象的某個(gè)屬性,該引用并不會(huì)改變。所以點(diǎn)擊事件中的作用域鏈?zhǔn)冀K訪問(wèn)的都是同一個(gè)vRef

react中useEffect閉包問(wèn)題怎么解決

5. 將v換成對(duì)象類型

其實(shí)和使用useRef一樣,只要是對(duì)象,僅修改某個(gè)屬性也不會(huì)改變?cè)搒tate所指向的地址。

關(guān)于“react中useEffect閉包問(wèn)題怎么解決”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí),可以關(guān)注億速云行業(yè)資訊頻道,小編每天都會(huì)為大家更新不同的知識(shí)點(diǎn)。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI