【react 高频面试题—核心原理篇】:useEffect 的依赖项如果是数组或对象(引用类型),会有什么问题?如何解决?

面试官提问:useEffect 的依赖项如果是数组或对象(引用类型),会有什么问题?如何解决?

💡 核心回答技巧

这道题考察的是两个核心点:

  1. React 依赖检查机制 :使用的是 Object.is(浅比较)。
  2. 闭包陷阱与死循环:引用类型地址改变导致的不必要渲染或无限循环。

1. 问题的本质:浅比较(Shallow Comparison)

React 在每次渲染后会对比 useEffectdeps 数组。 对于基本类型(string, number, boolean),比较的是 。 对于引用类型(Object, Array, Function),比较的是内存地址

典型错误场景:

JavaScript

scss 复制代码
function UserProfile() {
  const [count, setCount] = useState(0);
  
  // 每次组件重新渲染,options 都会被重新创建,指向一个新的内存地址
  const options = { id: 1 }; 

  useEffect(() => {
    console.log('执行了!');
  }, [options]); // 即使 options 的内容没变,地址变了,useEffect 就会触发
}

如果 useEffect 内部又触发了状态更新,就会导致无限死循环

2. 解决方案有哪些?

方案一:使用 useMemouseCallback

这是最正统的方法,将引用类型"持久化"。

  • useMemo:缓存对象/数组。
  • useCallback:缓存函数。

JavaScript

scss 复制代码
const options = useMemo(() => ({ id: 1 }), []); 
// 只要依赖项不变,options 的引用地址在多次渲染间保持不变

方案二:属性拆解(推荐)

如果对象很大,但我只关心其中的某个属性,直接把该属性放入依赖项。

JavaScript

scss 复制代码
useEffect(() => {
  // 业务逻辑
}, [options.id]); // 监听基本类型 id,而不是整个对象

方案三:使用 useRef

如果这个变量不需要参与渲染(即它的改变不需要触发 UI 更新),可以用 useRef 存储。useRef 返回的是同一个对象引用。

方案四:对于函数,写在 useEffect 内部

如果函数只在效应内部使用,直接移进去,避免将其作为依赖。


3. 进阶追问:如果不小心造成了死循环,该如何排查?

  • React DevTools:查看哪个 Hook 的数据一直在跳动。
  • 日志大法 :在 useEffect 第一行打 console.log,看触发频率。

🌟 总结

"useEffect 的依赖项对比执行的是浅比较 。对于引用类型,即使内容一致,只要内存地址改变,效应就会重新执行。解决办法通常是使用 useMemo/useCallback 冻结引用 ,或者拆解属性,确保只有当真正相关的数据变化时,才触发逻辑。"

相关推荐
Live0000019 小时前
在鸿蒙中使用 Repeat 渲染嵌套列表,修改内层列表的一个元素,页面不会更新
前端·javascript·react native
柳杉19 小时前
使用Ai从零开发智慧水利态势感知大屏(开源)
前端·javascript·数据可视化
兆子龙19 小时前
从高阶函数到 Hooks:React 如何减轻开发者的心智负担(含 Demo + ahooks 推荐)
前端
狗胜19 小时前
测试文章 - API抓取
前端
三小河19 小时前
VS Code 集成 claude-code 教程:告别海外限制,无缝对接国内大模型
前端·程序员
jerrywus19 小时前
前端老哥的救命稻草:用 Obsidian 搞定 Claude Code 的「金鱼记忆」
前端·agent·claude
球球pick小樱花20 小时前
游戏官网前端工具库:海内外案例解析
前端·javascript·css
用户605723748730820 小时前
AI 编码助手的规范驱动开发 - OpenSpec 初探
前端·后端·程序员
狗胜20 小时前
AI观察日记 2026-03-02|CLAUDE、TYPE、APPFUNCTIONS:掘金热门里的下一步信号
前端