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 的状态更新流程。

相关推荐
西洼工作室1 小时前
前端Token失效检测与自动登出机制详解
前端
李剑一1 小时前
华为二面稳了!面试官:请详细说明一下前端性能指标(FCP/LCP/CLS/TTI/TBT),如何采集、解读与优化?
前端·面试
KaMeidebaby7 小时前
卡梅德生物技术快报|骆驼纳米抗体:从原核表达、高通量测序到分子对接全流程实现
前端·数据库·其他·百度·新浪微博
子兮曰9 小时前
Node.js v26.1.0 深度解读:FFI、后量子密码与调试器的进化
前端·后端·node.js
测试员周周10 小时前
【Appium 系列】第06节-页面对象实现 — LoginPage 实战
开发语言·前端·人工智能·python·功能测试·appium·测试用例
西洼工作室11 小时前
前端直传OSS服务端签名(Policy+Signature)/STS临时凭证
前端·文件上传·oss
你很易烊千玺11 小时前
日常练习-数组 字符串常用的场景
前端·javascript·字符串·数组
weixin1997010801612 小时前
[特殊字符] RESTful API 接口规范详解:构建高效、可扩展的 Web 服务(附 Python 源码)
前端·python·restful