React第十七章(useRef)

useRef

當你在React中需要處理DOM元素或需要在組件渲染之間保持持久性數據時,便可以使用useRef。

import { useRef } from 'react';
const refValue = useRef(initialValue)
refValue.current // 訪問ref的值 類似于vue的ref,Vue的ref是.value,其次就是vue的ref是響應式的,而react的ref不是響應式的

通過Ref操作DOM元素

參數
  • initialValue:ref 對象的 current 屬性的初始值。可以是任意類型的值。這個參數在首次渲染后被忽略。
返回值
  • useRef返回一個對象,對象的current屬性指向傳入的初始值。 {current:xxxx}
注意
  • 改變 ref.current 屬性時,React 不會重新渲染組件。React 不知道它何時會發生改變,因為 ref 是一個普通的 JavaScript 對象。
  • 除了 初始化 外不要在渲染期間寫入或者讀取 ref.current,否則會使組件行為變得不可預測。
import { useRef } from "react"
function App() {//首先,聲明一個 初始值 為 null 的 ref 對象let div = useRef(null)const heandleClick = () => {//當 React 創建 DOM 節點并將其渲染到屏幕時,React 將會把 DOM 節點設置為 ref 對象的 current 屬性console.log(div.current)}return (<>{/*然后將 ref 對象作為 ref 屬性傳遞給想要操作的 DOM 節點的 JSX*/}<div ref={div}>dom元素</div><button onClick={heandleClick}>獲取dom元素</button></>)
}
export default App

數據存儲

我們實現一個保存count的新值和舊值的例子,但是在過程中我們發現一個問題,就是num的值一直為0,這是為什么呢?

因為等useStateSetCount執行之后,組件會重新rerender,num的值又被初始化為了0,所以num的值一直為0。

import React, { useLayoutEffect, useRef, useState } from 'react';function App() {let num = 0let [count, setCount] = useState(0)const handleClick = () => {setCount(count + 1)num = count;};return (<div><button onClick={handleClick}>增加</button><div>{count}:{num}</div></div>);
}export default App;

在這里插入圖片描述

如何修改?

我們可以使用useRef來解決這個問題,因為useRef只會在初始化的時候執行一次,當組件reRender的時候,useRef的值不會被重新初始化。

import React, { useLayoutEffect, useRef, useState } from 'react';function App() {let num = useRef(0)let [count, setCount] = useState(0)const handleClick = () => {setCount(count + 1)num.current = count;};return (<div><button onClick={handleClick}>增加</button><div>{count}:{num.current}</div></div>);
}export default App;

在這里插入圖片描述

實際應用

我們實現一個計時器的例子,在點擊開始計數的時候,計時器會每300ms執行一次,在點擊結束計數的時候,計時器會被清除。

問題

我們發現,點擊end的時候,計時器并沒有被清除,這是為什么呢?

原因

這是因為組件一直在重新ReRender,所以timer的值一直在被重新賦值為null,導致無法清除計時器。

import React, { useLayoutEffect, useRef, useState } from 'react';function App() {console.log('render')let timer: NodeJS.Timeout | null = nulllet [count, setCount] = useState(0)const handleClick = () => {timer = setInterval(() => {setCount(count => count + 1)}, 300)};const handleEnd = () => {console.log(timer);if (timer) {clearInterval(timer)timer = null}};return (<div><button onClick={handleClick}>開始計數</button><button onClick={handleEnd}>結束計數</button><div>{count}</div></div>);
}export default App;
如何修改?

我們可以使用useRef來解決這個問題,因為useRef的值不會因為組件的重新渲染而改變。

import React, { useLayoutEffect, useRef, useState } from 'react';function App() {console.log('render')let timer = useRef<null | NodeJS.Timeout>(null)let [count, setCount] = useState(0)const handleClick = () => {timer.current = setInterval(() => {setCount(count => count + 1)}, 300)};const handleEnd = () => {if (timer.current) {clearInterval(timer.current)timer.current = null}};return (<div><button onClick={handleClick}>開始計數</button><button onClick={handleEnd}>結束計數</button><div>{count}</div></div>);
}export default App;

注意事項

  1. 組件在重新渲染的時候,useRef的值不會被重新初始化。

  2. 改變 ref.current 屬性時,React 不會重新渲染組件。React 不知道它何時會發生改變,因為 ref 是一個普通的 JavaScript 對象。

  3. useRef的值不能作為useEffect等其他hooks的依賴項,因為它并不是一個響應式狀態。

  4. useRef不能直接獲取子組件的實例,需要使用forwardRef。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/62422.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/62422.shtml
英文地址,請注明出處:http://en.pswp.cn/web/62422.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

【C++】內存分布、new、delete、 operator new、operator delete

內存分布 在C語言和C中&#xff0c;程序內存被劃分成六個部分&#xff1a; 內核空間、棧、內存映射段、堆、數據段、代碼段 棧&#xff1a;又稱堆棧&#xff0c;主要為非靜態局部變量、函數參數、返回值等&#xff0c;棧的生長方向是向下生長的 內存映射段&#xff1a;高效的…

代碼隨想錄算法訓練營day37|動態規劃part5

今天的幾道題目都比較簡單&#xff0c;思路也比較相似&#xff0c;都是利用完全背包。完全背包和01背包的不同點在于完全背包每個元素可以取多次&#xff0c;而01背包只能取1次&#xff0c;所以在dp一維數組遍歷時&#xff0c;完全背包仍然要從前往后遍歷&#xff0c;并且無論是…

混合云策略在安全領域受到青睞

Genetec 發布了《2025 年物理安全狀況報告》&#xff0c;該報告根據超過 5,600 名該領域領導者&#xff08;其中包括 100 多名來自澳大利亞和新西蘭的領導者&#xff09;的回應&#xff0c;揭示了物理安全運營的趨勢。 報告發現&#xff0c;澳大利亞和新西蘭的組織采用混合云策…

FastAPI解決跨域報錯net::ERR_FAILED 200 (OK)

目錄 一、跨域問題的本質 二、FastAPI中的CORS處理 1. 安裝FastAPI和CORS中間件 2. 配置CORS中間件 3. 運行FastAPI應用 三、解決跨域報錯的步驟 四、案例:解決Vue.js與FastAPI的跨域問題 1. Vue.js前端應用 2. FastAPI后端API 3. 配置CORS中間件 4. 運行和測試 五…

為什么 JavaScript 中的 `new` 運算符報錯?

在 JavaScript 中&#xff0c;new 運算符通常用于創建一個新對象并調用構造函數來初始化對象。然而&#xff0c;new 運算符可能會引發一些錯誤&#xff0c;通常是由于以下原因導致的&#xff1a; 構造函數沒有正確的定義&#xff1a; 如果使用 new 運算符調用的函數沒有正確地定…

霍爾效應電流傳感器

適用于大電流&#xff0c;低功耗的電流檢測&#xff1a; TVS選型: RS232的隔離電路: 單片機采集200伏高壓 如何做隔離電路&#xff1a; 采用線性光電耦合器HCNR200實現高壓直流母線電壓的精確采樣。還是用電阻分壓&#xff0c;只是在ADC檢測階段加上隔離芯片&#xff1a;

如何設置Java爬蟲的異常處理?

在Java爬蟲中設置異常處理是非常重要的&#xff0c;因為網絡請求可能會遇到各種問題&#xff0c;如連接超時、服務器錯誤、網絡中斷等。通過合理的異常處理&#xff0c;可以確保爬蟲的穩定性和健壯性。以下是如何在Java爬蟲中設置異常處理的步驟和最佳實踐&#xff1a; 1. 使用…

ceph /etc/ceph-csi-config/config.json: no such file or directory

環境 rook-ceph 部署的 ceph。 問題 kubectl describe pod dragonfly-redis-master-0Warning FailedMount 7m59s (x20 over 46m) kubelet MountVolume.MountDevice failed for volume "pvc-c63e159a-c940-4001-bf0d-e6141634cc55" : rpc error: cod…

【計網筆記】習題

物理層 不屬于物理層接口規范定義范疇的是&#xff08;C&#xff09; A. 接口形狀 B. 引腳功能 C. 物理地址 D. 信號電平 【2023-912】光網絡只能通過導向型介質傳播。&#xff08;&#xff09; 【2017-408】若信道在無噪聲情況下的極限數據傳輸速率不小于信噪比為30dB條件下的…

最新 AI 編程工具全面對比:v0、Bolt.new、Cursor、Windsurf

隨著人工智能的快速發展&#xff0c;越來越多的 AI 驅動的開發工具應運而生&#xff0c;旨在提升開發效率、優化開發流程&#xff0c;并減輕開發者的工作負擔。在這個背景下&#xff0c;四款新興的 AI 編程工具&#xff1a;v0、Bolt.new、Cursor 和 Windsurf&#xff0c;各具特…

【C++算法】35.位運算_兩整數之和

文章目錄 題目鏈接&#xff1a;題目描述&#xff1a;解法C 算法代碼&#xff1a; 題目鏈接&#xff1a; 371. 兩整數之和 題目描述&#xff1a; 解法 筆試的話直接 return ab&#xff1b; 接下來講一下這題的解法&#xff1a; 位運算&#xff08;異或運算-無進位相加&#xff…

PyCharm+Selenium+Pytest配置小記

1、下載ChromeDriver&#xff1a; Chrome130以后的Driver下載&#xff1a; Chrome for Testing availabilityhttps://googlechromelabs.github.io/chrome-for-testing/ &#xff08;1&#xff09;查看自己Crome瀏覽器的版本&#xff1a;設置-->關于 Chrome&#xff1b; &…

【C++】虛函數

類中聲明函數成員的時候&#xff0c;在函數的前面加上virtual關鍵字&#xff0c;則該成員為虛函數 虛函數的特點 如果在類中定義的虛函數&#xff0c;那么系統會為這個類維護一個虛函數表類中會多出4個字節的指針去指向這個虛函數表&#xff0c;在虛函數表中保存了虛函數的首…

如何在UI自動化測試中創建穩定的定位器?

如何在UI自動化測試中創建穩定的定位器&#xff1f; 前言1. 避免使用絕對路徑2. 避免在定位器中使用索引3. 避免多個類名的定位器4. 避免動態和自動生成的ID5. 確保定位器唯一6. 處理隱藏元素的策略7. 謹慎使用基于文本的定位器8. 使用AI創建穩定的定位器 總結 前言 在自動化測…

明日宣講 香港科技大學廣州|數據科學與分析學域碩博招生宣講會【湖南大學專場】

&#x1f4e3;&#x1f4e3;&#x1f4e3;明日宣講 香港科技大學廣州&#xff5c;數據科學與分析學域碩博招生宣講會【湖南大學專場】 時間&#xff1a;2024年12月10日&#xff08;星期二&#xff09;15:00 地點&#xff1a;湖南大學信息科學與工程學院624會議室 報名鏈接&…

BurpSuite-8(FakeIP與爬蟲審計)

聲明&#xff1a;學習視頻來自b站up主 瀧羽sec&#xff0c;如涉及侵權馬上刪除文章 感謝瀧羽sec 團隊的教學 視頻地址&#xff1a;IP偽造和爬蟲審計_嗶哩嗶哩_bilibili 一、FakeIP 1.配置環境 BurpSuite是java環境下編寫的&#xff0c;而今天的插件是python編寫的&#xff0c…

2024 一帶一路暨金磚國家技能發展與技術創新大賽【網絡安全防護治理實戰技能賽項】樣題(中職組)

2024 一帶一路暨金磚國家技能發展與技術創新大賽【網絡安全防護治理實戰技能賽項】樣題&#xff08;中職組&#xff09; 1.基礎設置和安全強化&#xff08;xxx 分&#xff09;1.3. 任務內容: 2.安全監測和預警&#xff08;xxx 分&#xff09;2.1. 任務一&#xff1a;建立目錄安…

Qt C++ 顯示多級結構體,包括結構體名、變量名和值

文章目錄 mainwindow.hmainwindow.cppstructures.hmain.cpp QTreeView 和 QStandardItemModel 來實現。以下是實現這一功能的步驟和示例代碼&#xff1a; 定義多級結構體&#xff1a; 假設你有一個多級結構體&#xff0c;如下所示&#xff1a; struct SubStruct {int subValue…

CISC和RISC的區別

CISC是什么 CISC是一種指令集設計方式&#xff0c;全程叫復雜指令集&#xff0c;這種指令設計模式包含大量的復雜指令和復雜的尋址方式&#xff0c; 以下是CISC架構的一些主要特點&#xff1a; 豐富的指令集&#xff1a;CISC處理器通常包含大量的指令&#xff0c;這些指令覆蓋…

20 設計模式之職責鏈模式(問題處理案例)

一、什么是職責鏈模式 職責鏈模式是一種行為型設計模式&#xff0c;它允許將請求沿著處理者的鏈進行傳遞&#xff0c;直到有一個處理者能夠處理它為止。換句話說&#xff0c;它將請求的發送者和接收者解耦&#xff0c;使得多個對象都有機會處理這個請求&#xff0c;從而避免了將…