react的useRef 作用:获取DOM、保存可变数据、区别 createRef

useRef 是 React 中非常常用的 Hook,核心特点:

在组件多次渲染之间保存一个可变值,并且修改它不会触发重新渲染。

主要有三个用途:获取 DOM、保存可变数据、和 createRef 的区别。


1. 获取 DOM 元素(最常见)

类似 Vue 的 ref

例子:获取 input 并自动聚焦

复制代码
import { useRef, useEffect } from "react";

function App() {
  const inputRef = useRef(null);

  useEffect(() => {
    inputRef.current.focus();
  }, []);

  return (
    <input ref={inputRef} />
  );
}

执行流程:

复制代码
首次渲染
↓
inputRef = { current: null }

DOM挂载完成
↓
React自动赋值
inputRef.current = input节点

useEffect执行
↓
focus()

此时:

复制代码
inputRef.current

得到:

复制代码
<input>

常见用途:

复制代码
获取元素高度
获取滚动位置
focus()
播放视频
canvas操作
上传文件

例如:

复制代码
divRef.current.scrollIntoView();
videoRef.current.play();

2. 保存可变数据(不会触发重新渲染)

这是很多人忽略但很重要的用途。

假设统计点击次数:

用 state

复制代码
const [count, setCount] = useState(0);

修改:

复制代码
setCount(count + 1);

会重新渲染。


用 ref

复制代码
const countRef = useRef(0);

function add() {
    countRef.current++;
    console.log(countRef.current);
}

修改:

复制代码
countRef.current++;

不会重新 render。

适合保存:

复制代码
定时器ID
上一次值
缓存数据
防抖timer
socket实例
请求状态

例:

保存 setTimeout:

复制代码
const timerRef = useRef(null);

useEffect(() => {
  timerRef.current = setTimeout(() => {
      console.log("执行");
  },1000);

  return () => {
      clearTimeout(timerRef.current);
  }
},[])

避免:

复制代码
组件销毁
timer丢失
内存泄漏

3. 获取上一次值(经典面试题)

记录前一个 state:

复制代码
function App() {
  const [count,setCount]=useState(0);
  const prevRef=useRef();

  useEffect(()=>{
      prevRef.current=count;
  },[count])

  return (
      <>
          当前:{count}
          上次:{prevRef.current}
      </>
  )
}

效果:

复制代码
当前:3
上次:2

因为:

复制代码
render
↓
useEffect
↓
更新 ref

4. useRef 与 createRef 区别(面试高频)

createRef

复制代码
class App extends React.Component {
    inputRef = React.createRef();
}

主要给 类组件 用。

函数组件里:

复制代码
const ref = createRef();

问题:

每次 render 都会重新创建:

复制代码
render1
ref = {}

render2
ref = {}

不是同一个对象。


useRef

复制代码
const ref = useRef();

只创建一次:

复制代码
render1
ref={current:null}

render2
还是同一个ref

不会变。

所以函数组件推荐:

复制代码
函数组件 → useRef
类组件 → createRef

对比表:

区别 useRef createRef
适用 函数组件 类组件
是否重新创建
保存值 可以 不适合
获取DOM 支持 支持
触发渲染 不触发 不触发

总结一句:

复制代码
useRef =
获取DOM
+ 保存跨渲染可变值
+ 不触发更新
+ 函数组件替代 createRef

前端面试里经常问:

useRef 为什么修改 current 不会触发更新?

因为 React 不会追踪 ref.current 的变化,它只是普通对象:

复制代码
{
   current: xxx
}

修改对象属性,不会进入 React 的状态更新流程。

相关推荐
爱勇宝7 小时前
大多数人不是在使用 AI 赚钱,而是在帮 AI 公司赚钱
前端·后端·程序员
冬奇Lab8 小时前
每日一个开源项目(第143篇):page-agent - 纯 JS 的网页 GUI Agent,无需截图、无需插件、无需后端
前端·人工智能·agent
To_OC9 小时前
LC 994 腐烂的橘子:人人都说是 BFS 入门题,我却写了三遍才过
javascript·算法·leetcode
IT_陈寒12 小时前
React的这个渲染问题连官方文档都没说清楚
前端·人工智能·后端
追逐时光者14 小时前
别再满网找零散工具了,腾讯 QQ 浏览器这个“帮小忙”工具箱真能省时间
前端·后端
To_OC15 小时前
LC 200 岛屿数量:经典 DFS 入门题,我第一次写居然连方向都搞错了
javascript·算法·leetcode
Asmewill16 小时前
grep&curl命令学习笔记
前端
stringwu16 小时前
Flutter 开发必备:MVI 架构的高效实现指南
前端·flutter
用户21366100357217 小时前
Vue2组件化开发与父子通信
前端·vue.js
Momo__17 小时前
TypeScript satisfies 操作符——比 as 更安全的类型守门员
前端·typescript