【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 冻结引用 ,或者拆解属性,确保只有当真正相关的数据变化时,才触发逻辑。"

相关推荐
happymaker06267 分钟前
web前端学习日记——DAY04
前端·学习
发现一只大呆瓜13 分钟前
React-路由监听 / 跳转 / 守卫全攻略(附实战代码)
前端·react.js·面试
swipe1 小时前
为什么 RAG 一定离不开向量检索:从文档向量化到语义搜索的工程实现
前端·llm·agent
OpenTiny社区1 小时前
AI-Extension:让 AI 真的「看得到、动得了」你的浏览器
前端·ai编程·mcp
IT_陈寒1 小时前
Redis缓存击穿:3个鲜为人知的防御策略,90%开发者都忽略了!
前端·人工智能·后端
农夫山泉不太甜3 小时前
Tauri v2 实战代码示例
前端
消失的旧时光-19433 小时前
Android 面试高频:JSON 文件、大数据存储与断电安全(从原理到工程实践)
android·面试·json
yuhaiqiang3 小时前
被 AI 忽悠后,开始怀念搜索引擎了?
前端·后端·面试
红色石头本尊3 小时前
1-umi-前端工程化搭建
前端
真夜3 小时前
关于对echart盒子设置百分比读取的宽高没有撑开盒子解决方案
前端