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

相关推荐
AskHarries4 小时前
Workspace:文件系统、项目上下文和执行边界
java·服务器·前端
Aphasia3114 小时前
从内存模型看深浅拷贝
前端·javascript·面试
IT策士5 小时前
第45篇 k8s之实战:将 Web 应用迁移到 Kubernetes(下)
前端·容器·kubernetes
云水一下5 小时前
TypeScript 从零基础到精通(二):基础类型与类型系统
javascript·typescript
你怎么知道我是队长5 小时前
CRC校验C语言实现-CRC8、CRC16、CRC16的直接计算法、查表法
c语言·前端·javascript
Rain5095 小时前
mini-cc 终端 UI:用 React 写 CLI 是什么体验
前端·人工智能·react.js·ui·架构·前端框架·ai编程
wu8587734575 小时前
向量数据库不是银弹:从枚举漏检到 ReACT 多轮召回的实践路径
前端·数据库·react.js
meilindehuzi_a5 小时前
深入理解 JavaScript 执行机制:从编译阶段到调用栈底层实现
开发语言·javascript·ecmascript
古怪今人5 小时前
[前端]HTML盒模型与尺寸,标准文档流,块级元素、内联元素和行内块,CSS选择器
前端·css
小雨下雨的雨6 小时前
基于鸿蒙PC Electron框架技术完成的表单验证技术详解
前端·javascript·华为·electron·前端框架·鸿蒙