React面试题,useRef和普通变量的区别

在React中,useRef 和普通的变量(如 letconst)在行为和用途上有很大的不同。以下是 skipFirst = useRef(true) 和直接使用 let skipFirst = true 的效果对比:

useRef(true) 的效果

tsx 复制代码
const skipFirst = useRef(true);
  1. 持久性

    • useRef 创建的引用对象在组件的整个生命周期内保持不变。也就是说,skipFirst.current 的值会在每次组件渲染时保持一致,不会因为组件的重新渲染而丢失。
    • 例如,如果你在第一次渲染时将 skipFirst.current 设置为 false,那么在后续的渲染中,skipFirst.current 仍然是 false
  2. 不触发重新渲染

    • 修改 useRef 的值(如 skipFirst.current = false)不会触发组件的重新渲染。这使得 useRef 非常适合存储不需要触发重新渲染的值。
  3. 跨渲染保持状态

    • useRef 的值在组件的每次渲染之间保持不变,这对于存储如计数器、标记等状态非常有用,尤其是在需要跨渲染保持状态的场景中。

直接使用 let skipFirst = true 的效果

tsx 复制代码
let skipFirst = true;
  1. 非持久性

    • 使用 letconst 定义的变量在每次组件渲染时都会重新初始化。也就是说,每次组件重新渲染时,skipFirst 都会被重新设置为 true
    • 例如,如果你在第一次渲染时将 skipFirst 设置为 false,那么在组件重新渲染时,skipFirst 仍然会被重新初始化为 true
  2. 触发重新渲染

    • 修改普通变量的值(如 skipFirst = false)不会直接触发组件的重新渲染,但这种变量的值在每次渲染时都会被重新初始化,因此无法保持状态。
  3. 无法跨渲染保持状态

    • 普通变量在每次组件渲染时都会被重新初始化,无法保持跨渲染的状态。这使得它们不适合用于存储需要跨渲染保持的状态。

示例对比

假设我们有一个组件,需要在第一次渲染时跳过某个副作用,但后续渲染时执行该副作用。我们分别使用 useRef 和普通变量来实现:

使用 useRef
tsx 复制代码
import { useEffect, useRef } from 'react';

function MyComponent() {
  const skipFirst = useRef(true);

  useEffect(() => {
    if (skipFirst.current) {
      skipFirst.current = false;
      return;
    }
    console.log('This effect runs after the first render');
  }, []);

  return <div>My Component</div>;
}
  • 效果skipFirst.current 在第一次渲染时为 true,跳过副作用。后续渲染时,skipFirst.current 保持为 false,副作用会执行。
使用普通变量
tsx 复制代码
import { useEffect } from 'react';

function MyComponent() {
  let skipFirst = true;

  useEffect(() => {
    if (skipFirst) {
      skipFirst = false;
      return;
    }
    console.log('This effect runs after the first render');
  }, []);

  return <div>My Component</div>;
}
  • 效果 :每次组件重新渲染时,skipFirst 都会被重新初始化为 true,因此副作用永远不会执行。

总结

  • useRef:适合存储需要跨渲染保持状态的值,修改值不会触发重新渲染。
  • 普通变量(let/const:每次组件重新渲染时都会被重新初始化,无法保持跨渲染的状态。

在上述场景中,使用 useRef 是正确的选择,因为它可以确保 skipFirst 的值在组件的整个生命周期内保持一致,从而实现跳过第一次渲染的效果。

相关推荐
SuperEugene1 分钟前
表单最佳实践:从 v-model 到自定义表单组件(含校验)
前端·javascript·vue.js
昨晚我输给了一辆AE861 分钟前
为什么现在不推荐使用 React.FC 了?
前端·react.js·typescript
不会敲代码12 分钟前
深入浅出 React 闭包陷阱:从现象到原理
前端·react.js
不会敲代码14 分钟前
React性能优化:深入理解useMemo和useCallback
前端·javascript·react.js
Dilettante2586 分钟前
我的 Monorepo 实践经验:从基础概念到最佳实践
前端·前端工程化
只会cv的前端攻城狮24 分钟前
Elpis-Core — 融合 Koa 洋葱圈模型实现服务端引擎
前端·后端
Java小卷1 小时前
流程设计器为啥选择diagram-js
前端·低代码·工作流引擎
HelloReader1 小时前
Isolation Pattern(隔离模式)在前端与 Core 之间加一道“加密网关”,拦截与校验所有 IPC
前端
兆子龙2 小时前
从 float 到 Flex/Grid:CSS 左右布局简史与「刁钻」布局怎么搞
前端·架构
YukiMori232 小时前
一个有趣的原型继承实验:为什么“男人也会生孩子”?从对象赋值到构造函数继承的完整推演
前端·javascript