react中useRef的應(yīng)用使用詳解
何為useRef
useRef是隨著react函數(shù)式組件發(fā)展而來的,是react眾多官方hook中的一個(gè),調(diào)用useRef可以返回一個(gè)伴隨這組件整個(gè)聲明周期不發(fā)生改變的對(duì)象,這個(gè)對(duì)象常見的用途有兩個(gè):
- 用于綁定dom元素,從而實(shí)現(xiàn)對(duì)dom元素的操作
- 用于保存不希望隨著組件重新渲染而改變的值,如定時(shí)器
在項(xiàng)目中的應(yīng)用
我在做自己的網(wǎng)站時(shí)遇到過一些場(chǎng)景,實(shí)現(xiàn)效果與預(yù)想中的不一致,讓我頭大了好一陣子,所幸最終都得到了解決,為了在以后能夠回來看一下,同時(shí)記錄學(xué)習(xí)過程中的一些知識(shí)點(diǎn),現(xiàn)將這兩個(gè)場(chǎng)景記錄如下。
一、編寫自己的防抖函數(shù)
在編寫登錄注冊(cè)功能時(shí),需要對(duì)用戶輸入的信息進(jìn)行校驗(yàn),包括前端的校驗(yàn)以及和后端通信的校驗(yàn),如果只是前端的校驗(yàn)還好,如果在用戶輸入的信息每次發(fā)生變化都去和服務(wù)器通信校驗(yàn)的話,大量的http請(qǐng)求會(huì)對(duì)服務(wù)器造成壓力,所以我希望對(duì)這些行為進(jìn)行防抖處理。
useDebounce.ts
import { useEffect, useRef } from "react"
export const useDebounce = (f:Function,delay:number)=>{
//使用useRef保存計(jì)時(shí)器,以此確保在組件更新時(shí)始終是同一個(gè)計(jì)時(shí)器,而不是重新創(chuàng)建
const {current} = useRef<{timmer:any}>({timmer:null})
useEffect(()=>{
return ()=>{
//組件銷毀時(shí)清除計(jì)時(shí)器
clearTimeout(current.timmer)
}
},[])
return function(...args:any[]){
//如果計(jì)時(shí)器存在,則清除該計(jì)時(shí)器
if(current.timmer){
clearTimeout(current.timmer)
}
//重新賦值計(jì)時(shí)器,并在計(jì)時(shí)結(jié)束后執(zhí)行回調(diào)函數(shù)
current.timmer = setTimeout(() => {
f.apply(useDebounce,args)
}, delay);
}
}這里我將防抖函數(shù)寫成了一個(gè)自定義的hook,也是寫的第一個(gè)hook,調(diào)用時(shí)需要傳入兩個(gè)參數(shù),第一個(gè)參數(shù)是需要進(jìn)行防抖處理的函數(shù),第二個(gè)參數(shù)是防抖的延時(shí)時(shí)間,hook返回值是經(jīng)過防抖處理的函數(shù)。
二、解決回調(diào)函數(shù)中獲取不到最新state值的問題
在實(shí)現(xiàn)獲取評(píng)論列表時(shí),我希望每次只獲取一定數(shù)量的評(píng)論,當(dāng)用戶瀏覽到頁面底端時(shí)再獲取新的評(píng)論,在實(shí)現(xiàn)過程中我用到了intersectionObserverAPI,并在它的回調(diào)函數(shù)中引用了組件中的一些state。
當(dāng)我直接在useEffect副作用函數(shù)中進(jìn)行綁定監(jiān)聽的元素時(shí),發(fā)現(xiàn)回調(diào)函數(shù)中的state值一直保持observer對(duì)象創(chuàng)建時(shí)的值,經(jīng)過查詢相關(guān)資料,得知這可能是因?yàn)殚]包的影響。解決辦法是使用useRef創(chuàng)建一個(gè)對(duì)象用于保存observer對(duì)象,并在相關(guān)state值發(fā)生變化后釋放原來的observer對(duì)象,創(chuàng)建一個(gè)新的observer對(duì)象,并重新綁定要監(jiān)聽的元素。
//指向要監(jiān)聽的元素
const bref = useRef(null)
//保存observer對(duì)象
const observer = useRef<any>()
//每當(dāng)comments變化,都會(huì)重新創(chuàng)建一個(gè)observer對(duì)象,其回調(diào)函數(shù)中引用的就是最新的state值
useEffect(() => {
let c = new IntersectionObserver((entries) => {
if (entries[0].intersectionRatio > 0) {
if (page * pageNum <= count) {
const fd = new FormData()
fd.append('articleId', param.articleid)
fd.append('page', page + 1)
fd.append('pageNum', pageNum)
http({ url: '/comment/comments', options: { method: 'POST', body: fd } }).then(res => {
if (count !== res.count) {
setcount(res.res.count)
}
setpage(page + 1)
setcomments([...comments, ...res.res.rows])
})
}
}
})
//取消舊的監(jiān)聽
if (observer.current) {
observer.current.unobserve(bref.current)
}
//保存新的observer對(duì)象,并建立新的監(jiān)聽
observer.current = c
observer.current.observe(bref.current)
}, [comments])
實(shí)踐出真知,在項(xiàng)目中發(fā)現(xiàn)問題,尋找解決問題的方法,大江不止兮水長(zhǎng)流,不斷積累,豐富閱歷,提升能力。
到此這篇關(guān)于react中useRef的應(yīng)用使用詳解的文章就介紹到這了,更多相關(guān)react中useRef的應(yīng)用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用React18和WebSocket構(gòu)建實(shí)時(shí)通信功能詳解
WebSocket是一種在Web應(yīng)用中實(shí)現(xiàn)雙向通信的協(xié)議,它允許服務(wù)器主動(dòng)向客戶端推送數(shù)據(jù),而不需要客戶端發(fā)起請(qǐng)求,本文將探索如何在React?18應(yīng)用中使用WebSocket來實(shí)現(xiàn)實(shí)時(shí)通信,感興趣的可以了解下2024-01-01
使用React路由實(shí)現(xiàn)頁面導(dǎo)航的示例代碼
在構(gòu)建現(xiàn)代Web應(yīng)用程序時(shí),前端路由是一個(gè)不可或缺的部分,今天,我們將討論如何在React中使用React Router來實(shí)現(xiàn)頁面導(dǎo)航,在這篇博客中,我們將會(huì)探索路由的基本概念,設(shè)置React Router,并通過示例代碼來展示如何實(shí)現(xiàn)復(fù)雜的頁面導(dǎo)航,需要的朋友可以參考下2025-02-02
React+TS+IntersectionObserver實(shí)現(xiàn)視頻懶加載和自動(dòng)播放功能
通過本文的介紹,我們學(xué)習(xí)了如何使用 React + TypeScript 和 IntersectionObserver API 來實(shí)現(xiàn)一個(gè)視頻播放控制組件,該組件具有懶加載功能,只有在用戶滾動(dòng)頁面且視頻進(jìn)入視口時(shí)才開始下載視頻資源,需要的朋友可以參考下2023-04-04
VSCode配置react開發(fā)環(huán)境的步驟
本篇文章主要介紹了VSCode配置react開發(fā)環(huán)境的步驟,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-12-12
react配置代理setupProxy.js無法訪問v3.0版本問題
這篇文章主要介紹了react配置代理setupProxy.js無法訪問v3.0版本問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07

