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

相关推荐
hibear2 小时前
Smart Ticker - 支持任意字符的高性能文本差异动画滚动组件
前端·vue.js·react.js
脱氧核糖核酸2 小时前
2026了你还只会写点prompt?从AI提示词到可控自动化的演进之路
前端
HabaraAi2 小时前
记一次发现 DataTransfer 的 getData 的有趣问题
前端
a17798877122 小时前
print.js打印
前端·javascript·html
小林攻城狮2 小时前
前端实时语音转写:原生 MediaRecorder API 实践
前端·vue.js
Sport2 小时前
用全会,问全废:CSS高频面试题
前端·javascript·面试
Maxkim2 小时前
「✍️JS原子笔记 」零基础吃透 Proxy 数据响应式
前端·javascript·面试
踏浪无痕2 小时前
Java 17 升级避坑:如何安全处理反射访问限制
后端·面试·架构
hashiqimiya2 小时前
vue前端打包配置后端代理
前端