一、核心机制

1.合成事件机制

1.设计目的与核心原理

1.跨浏览器一致性
  • 统一不同浏览器的事件处理接口(如event.target的行为)
  • 修复浏览器兼容性问题(如IE事件模型差异)
2.性能优化
  • 事件委托:将事件绑定到根节点(React 17+ 为应用根DOM),而非每个子元素
  • 事件池化Event Pooling):复用事件对象,减少内存开销
3.扩展能力
  • 支持自定义事件类型(如 onDoubleClick
  • 实现高级功能(如事件优先级调度)

2.事件委托机制演进

| ​React 版本​ | ​委托层级​ | ​核心变化​ | | --- | --- | --- | --- | | 16.x 及之前 | 所有事件委托到 document | 多 React 应用共存时事件可能冲突 | | 17.x 及之后 | 委托到应用根 DOM 节点 | 隔离不同 React 版本的事件系统,避免全局污染 | ```js |

const rootNode = document.getElementById(‘root’); ReactDOM.render(, rootNode);

 1#### 3.合成事件对象
 2
 3##### 1.核心属性
 4
 5```js
 6interface SyntheticEvent {
 7  nativeEvent: Event;          
 8  currentTarget: DOMElement;  
 9  target: DOMElement;         
10  type: string;               
11  isDefaultPrevented(): boolean;
12  isPropagationStopped(): boolean;
13  persist(): void;           
14}
2.事件池化示例
 1function handleClick(event) {
 2  
 3  setTimeout(() => {
 4    console.log(event.target); 
 5  }, 100);
 6  
 7  
 8  event.persist();
 9  setTimeout(() => {
10    console.log(event.target); 
11  }, 100);
12}

4.事件处理流程

1.事件注册
  • React初始化时注册所有支持的事件(如onClick,onChange
  • 通过EventListener在根节点监听原生事件
2.事件触发
 1原生事件触发根节点捕获事件React 生成 SyntheticEvent收集事件监听器按组件树冒泡/捕获顺序执行
3.执行顺序
  • 捕获阶段父组件 onClickCapture -> 子组件 onClickCapture
  • 冒泡阶段子组件 onClick -> 父组件 onClick

与原生事件交互

1.混合使用场景
 1useEffect(() => {
 2  const handleNativeClick = (e) => {
 3    console.log('原生事件触发');
 4  };
 5  document.addEventListener('click', handleNativeClick);
 6
 7  return () => {
 8    document.removeEventListener('click', handleNativeClick);
 9  };
10}, []);
11
12
13const handleReactClick = (e) => {
14  console.log('合成事件触发');
15  e.stopPropagation(); 
16};
2.执行顺序
 1原生事件捕获) → 原生事件目标) → React 事件捕获) → React 事件目标) → React 事件冒泡) → 原生事件冒泡

6.常见问题与解决方案

1.事件阻止传播失败
  • 问题:e.stopPropagation() 仅阻止React事件传播,不影响原生事件
  • 方案:同时阻止原生事件传播
 1const handleClick = (e) => {
 2  e.stopPropagation();
 3  e.nativeEvent.stopImmediatePropagation();
 4};
2.事件监听器性能优化
  • 避免在渲染时创建新函数:使用useCallback函数缓存
 1const handleClick = useCallback((e) => {  }, []);

7.高频面试题

1.为什么React不直接将事件绑定在元素上?
  • 事件委托减少内存占用,动态更新组件时无需重新绑定事件
2.合成事件和原生事件的区别
  • 合成事件跨浏览器统一行为
  • 原生事件直接操作DOM,无React抽象层
3.如何全局阻止React事件
  • 劫持根节点事件监听
 1document.getElementById('root').addEventListener('click', e => {
 2  e.stopImmediatePropagation();
 3}, true);

2.组件更新触发条件与渲染优化

1.组件更新触发条件

组件重新渲染的根本原因是组件状态或数据依赖发生变化,具体触发场景如下:

触发条件说明
State 变化组件内部 useState/useReducer/this.setState 更新状态
Props 变化父组件重新渲染导致传入的 props 值变化
Context 更新组件订阅的 Context 数据发生变更
父组件重新渲染即使子组件的 props 未变化,父组件渲染仍可能导致子组件重新渲染(默认行为)
强制更新类组件调用 this.forceUpdate()
Hooks 依赖变化useEffect/useMemo/useCallback 的依赖数组元素变更

2.React渲染机制核心原理

1.渲染流程
  • 触发更新 -> 生成虚拟DOM -> Diff算法比较 -> 确定DOM更新范围 -> 提交到真实DOM
2.协调(Reconciliation)策略
  • 树对比:仅对比同层级节点,时间复杂度O(n)
  • Key值优化:列表项使用key帮助React识别元素移动/复用

3.渲染优化策略与实践

1.避免不必要的父组件渲染
  • 场景:父组件状态变化导致所有子组件重新渲染
  • 优化方案:
 1function Parent() {
 2  return (
 3    <>
 4      <ExpensiveChild />
 5      <StateContainer /> // 将易变状态抽离
 6    </>
 7  );
 8}
2.组件自身渲染控制
  • 类组件
 1class MyComponent extends React.PureComponent { 
 2  shouldComponentUpdate(nextProps, nextState) {
 3    return !shallowEqual(this.props, nextProps); 
 4  }
 5}
  • 函数组件
 1const MemoizedComponent = React.memo(MyComponent, (prevProps, nextProps) => {
 2  return prevProps.id === nextProps.id; 
 3});
3.精细化Hooks使用
  • 缓存计算结果
 1const expensiveValue = useMemo(() => computeValue(a, b), [a, b]);
  • 稳定函数引用
 1const handleClick = useCallback(() => {
 2  
 3}, [a]);
  • 按需订阅Context
 1const value = useContextSelector(MyContext, v => v.requiredField);
4.列表渲染优化
  • 虚拟滚动:使用react-windowreact-virtualized
 1import { FixedSizeList as List } from 'react-window';
 2
 3<List height={600} itemSize={35} itemCount={1000}>
 4  {({ index, style }) => <div style={style}>Row {index}</div>}
 5</List>
  • key值策略
 1{items.map((item, index) => <Item key={index} />)}
 2
 3
 4{items.map(item => <Item key={item.id} />)}

4.高频面试题

1.为什么父组件更新会导致所有子组件渲染?如何避免?
  • react默认采用”render and diff”策略,使用React.memo/shouldComponentUpdate阻断无效更新
2.useMemo一定能提升性能吗?使用场景是什么?
  • 不一定。仅当计算开销大且稳定时使用,否则可能因依赖数组计算反增开销
3.如何优化Context引起的渲染?
  • 拆分多个Context
  • 使用useContextSelector按需订阅
 1const ThemeButton = () => {
 2  const theme = useContextSelector(ThemeContext, v => v.color);
 3  return <button style={{ color: theme }}>Submit</button>;
 4};
4.函数组件每次渲染都会创建新函数,如何避免传递新props?
  • 使用useCallback缓存函数引用
 1const handleSubmit = useCallback(() => {  }, [deps]);

6.性能优化法则

  1. 优先解决重复渲染问题:使用React DevTools Profiler定位关键路径
  2. 避免过早优化:只在性能瓶颈出现时实施优化
  3. 保持组件纯净:减少渲染过程中的副作用操作
  4. 控制渲染范围:使用children props阻断无关更新
 1<Layout>
 2  <StaticContent /> {}
 3</Layout>
 4
 5
 6function Layout({ children }) {
 7  const [state, setState] = useState();
 8  return <div>{children}</div>;
 9}

3.Hooks核心原理

1.Hooks的设计目标

  1. 逻辑复用:解决类组件中高阶组件(HOC)和Render Props的嵌套地狱问题
  2. 简化组件:告别this绑定和生命周期方法的分散逻辑
  3. 函数式优先:拥抱函数式编程范式,提升代码可预测性
  4. 渐进式升级:兼容现有组件,无需重写即可逐步迁移

2.核心原理

1.闭包与链表存储
  • 存储结构:Hooks数据存储在Fiber节点的memoizedState属性中,通过单向链表管理
  • 执行顺序依赖:Hooks调用顺序在每次渲染中必须严格一致(链表顺序不可变)
  • 闭包陷阱:每个Hooks闭包捕获当次渲染的props/state快照
 1const fiber = {
 2  memoizedState: {
 3    memoizedState: '状态值',    
 4    next: {                   
 5      memoizedState: [],      
 6      next: null
 7    }
 8  }
 9};
2.调度机制
  • 优先级调度:Hooks更新请求会被Scheduler模块根据优先级(Immediate/UserBlocking/Normal)排队处理
  • 批量更新:React自动合并多个setState调用,减少渲染次数

3.核心Hooks原理解析

1.useState
  • 存储结构[state, dispatchAction]存储在链表节点中
  • 更新触发:调用dispatchAction会创建更新对象,触发重新渲染
  • 异步更新:状态更新在下次渲染时生效(遵循批量更新原则)
 1function useState(initial){
 2    const fiber = getCurrentFiber();
 3    const hook = fiber.memorizedState?.isStateHook ? fiber.memoizedState : { memouzedState: initial, next: null };
 4    const dispatch = (action) => {
 5        const update = { action };
 6        hook.queue.push(update);
 7        scheduleWork(); 
 8    };
 9    return [hook.memoizedState, dispatch];
10}
2.useEffect
  • 依赖对比:使用浅比较(Object.is)判断依赖数组变化
  • 执行时机:在浏览器完成布局与绘制后异步执行(避免阻塞渲染)
  • 清理机制:返回的清理函数会在下次effect执行前或组件卸载前执行
 1function areDepsEqual(prevDeps, nextDeps) {
 2  if (!prevDeps || !nextDeps) return false;
 3  for (let i = 0; i < prevDeps.length; i++) {
 4    if (!Object.is(prevDeps[i], nextDeps[i])) return false;
 5  }
 6  return true;
 7}
3.useRef
  • 跨渲染存储:ref对象在组件生命周期内保持不变
  • 直接修改ref.current的修改不会触发重新渲染
 1function useRef(initialValue) {
 2  const ref = { current: initialValue };
 3  return useMemo(() => ref, []); 
 4}

4.Hooks规则的本质

1.为什么必须顶层调用?
  • 链表顺序依赖:Hooks的存储依赖调用顺序,条件语句会破坏链表结构
2.为什么只能在函数组件中使用?
  • Fiber关联:Hooks需要绑定当前组件的Fiber节点,普通函数无此上下文

5.闭包陷阱与解决方案

1.过期闭包问题
 1function Timer() {
 2  const [count, setCount] = useState(0);
 3
 4  useEffect(() => {
 5    const timer = setInterval(() => {
 6      setCount(count + 1); 
 7    }, 1000);
 8    return () => clearInterval(timer);
 9  }, []); 
10
11  
12  useEffect(() => {
13    const timer = setInterval(() => {
14      setCount(c => c + 1); 
15    }, 1000);
16    return () => clearInterval(timer);
17  }, []);
18}
2.解决方案
  • 函数式更新setState(c => c + 1)
  • 依赖数组精准化:确保所有依赖项被正确声明
  • useRef穿透闭包:通过ref访问最新值
 1const countRef = useRef(count);
 2countRef.current = count;

6.高频面试题

1.Hooks如何实现状态隔离?
  • 每个组件实例的Fiber节点维护独立的Hooks链表,相同组件不同实例互不影响
2.自定义Hook的本质是什么?
  • 将多个内置Hooks组合为可复用的逻辑单元,遵循Hooks规则
3.为什么useEffect的依赖数组是浅比较
  • 性能优化考量,深比较成本过高。复杂对象应使用useMemo稳定引用
4.useMemo/useCallback如何避免重复计算?
  • 通过依赖数组决定是否重新计算,引用稳定时返回缓存值
 1const memoValue = useMemo(() => compute(a), [a]);
 2const memoFn = useCallback(() => action(a), [a]);

7.性能优化策略

优化手段实现方式适用场景
精细化依赖数组确保依赖数组只包含必要变量所有 Hooks
状态提升将状态提升到父组件或 Context多组件共享状态
惰性初始 stateuseState(() => expensiveInit())初始值计算成本高时
批量更新unstable_batchedUpdates(React 18 自动)合并多次状态更新

4.常用Hooks

1.基础Hooks

1.useState
  • 作用:为函数组件添加状态管理能力
  • 使用场景:组件内部的状态管理
 1const [count, setCount] = useState(() => 0); 
 2const increment = () => setCount(prev => prev + 1);
  • 注意
    • 状态更新是异步的,连续调用会被合并
    • 复杂对象建议使用useReducer
2.useEffect
  • 作用:处理副作用(数据请求、DOM操作、订阅)
  • 生命周期映射
    • componentDidMount -> 依赖数组为空 []
    • componentDidUpdate -> 指定依赖项 [dep]
    • componentWillUnmount -> 返回清理函数
 1useEffect(() => {
 2  const subscription = props.source.subscribe();
 3  return () => subscription.unsubscribe(); 
 4}, [props.source]);
  • 注意
    • 默认在浏览器完成渲染后异步执行
    • 使用useLayoutEffect处理同步DOM操作
3.useContext
  • 作用:跨组件层级传递数据
 1const ThemeContext = createContext('light');
 2
 3function App() {
 4  return (
 5    <ThemeContext.Provider value="dark">
 6      <Toolbar />
 7    </ThemeContext.Provider>
 8  );
 9}
10
11function Toolbar() {
12  const theme = useContext(ThemeContext);
13  return <div>{theme}</div>;
14}
  • 优化:使用memo防止无关组件更新

2.性能优化Hooks

1.useMemo
  • 作用:缓存计算结果,避免重复计算
 1const expensiveValue = useMemo(() => compute(a, b), [a, b]);
  • 注意
    • 依赖数组引用,避免子组件无效渲染
    • 不应用于有副作用的操作
2.useCallback
  • 作用:缓存函数引用,避免子组件无效渲染
 1const handleSubmit = useCallback(() => {
 2  submitData(name);
 3}, [name]);
  • 等价写法
 1const memoizedFn = useMemo(() => () => submitData(name), [name]);
3.React.memo
  • 作用:浅比较props变化,阻止无效渲染
 1const MemoComponent = React.memo(Child, (prev, next) => {
 2  return prev.id === next.id; 
 3});

进阶Hooks

1.useReducer
  • 作用:复杂状态逻辑管理(类似Redux)
 1const initialState = { count: 0 };
 2function reducer(state, action) {
 3    switch(action.type) {
 4        case 'increment':
 5            return { count: state.count + 1 };
 6        default: return state;
 7    }
 8}
 9const [count, dispatch] = useReducer(reducer, initialState);
  • 优势:适合多状态联动或逻辑状态复杂的场景
2.useRef
  • 作用:
    • 访问DOM元素
    • 保存可变值(不触发渲染)
 1const inputRef = useRef();
 2useEffect(() => inputRef.current.focus(), []);
 3
 4
 5const prevCount = useRef(count);
 6useEffect(() => {
 7  prevCount.current = count;
 8});
3.useLayoutEffect
  • 作用:同步执行副作用,在浏览器绘制前完成DOM修改
  • 场景:测量DOM布局、同步样式调整
 1useLayoutEffect(() => {
 2  const width = divRef.current.offsetWidth;
 3  setWidth(width); 
 4}, []);

4.特殊场景Hooks

1.useImperativeHandle
  • 作用:自定义暴露给父组件的实例方法
 1const FancyInput = forwardRef((props, ref) => {
 2  const inputRef = useRef();
 3  useImperativeHandle(ref, () => ({
 4    focus: () => inputRef.current.focus()
 5  }));
 6  return <input ref={inputRef} />;
 7});
2.useDebugValue
  • 作用:在React开发中工具中显示自定义hook标签
 1function useFriendStatus() {
 2  const [isOnline] = useState(null);
 3  useDebugValue(isOnline ? 'Online' : 'Offline');
 4  return isOnline;
 5}

5.高频面试题

1.useEffect和useLayoutEffect的区别?
  • useEffect异步执行(不阻塞渲染)
  • useLayoutEffect同步执行(在DOM更新后,浏览器绘制前)
2.如何避免useEffect的无限循环?
  • 正确设置依赖数组
  • 使用useCallback/useMemo稳定引用
3.useMemo和React.memo的区别?
  • useMemo缓存值
  • React.memo缓存组件渲染结果

5.虚拟DOM Diff算法原理

1.核心设计思想

React的Diff算法基于两个核心假设,以O(n)时间复杂度完成树结构的高效比较

  1. 类型差异假设:不同类型的元素会生成不同的树结构,直接替换整个子树
  2. Key稳定性假设:通过key标识相同层级的元素是否可复用

2.分层对比策略

React采用逐层递归比较,但不会跨层级追踪节点变化

 1旧树<div>          新树<section>
 2        <A/>               <A/>
 3        <B/>               <B/>
 4      </div>            </section>
 5
 6处理逻辑发现 divsection 类型不同直接销毁整个 div 子树重建 section 子树

3.同层级节点比较

当父节点类型相同时,递归比较其他子节点

1.列表节点对比优化
  • 场景:动态列表项的顺序变化(增删、排序)
  • 策略:使用唯一的key标识元素身份,最小化移动操作
 1<ul>
 2  <li key="a">A</li>
 3  <li key="b">B</li>
 4</ul>
 5
 6
 7<ul>
 8  <li key="b">B</li>
 9  <li key="a">A</li>
10</ul>
11
12
2.无key列表的默认行为
  • 使用索引作为key:可能导致错误复用(如列表项内容变化但位置不变)
  • 性能陷阱:列表顺序变化时,索引变化导致不必要的重新渲染

4.元素类型处理

1.类型相同
  • 更新属性:仅修改变化的属性(如className、style)
  • 递归比较子节点:继续对比子元素
2.类型不同
  • 销毁旧子树:触发旧组件 componentWillUnmount
  • 创建新子树:触发新组件 constructor -> render -> componentDidMount

5.Diff算法步骤分解

1.根节点对比
  • 类型不同 -> 整树替换
  • 类型相同 -> 进入属性比较
2.属性更新
 1function updateDOMProperties(domElement, prevProps, nextProps) {
 2  
 3  Object.keys(prevProps).forEach(propName => {
 4    if (!nextProps.hasOwnProperty(propName)) {
 5      domElement.removeAttribute(propName);
 6    }
 7  });
 8  
 9  
10  Object.keys(nextProps).forEach(propName => {
11    if (prevProps[propName] !== nextProps[propName]) {
12      domElement.setAttribute(propName, nextProps[propName]);
13    }
14  });
15}
3.子节点递归对比
  • 策略:双指针遍历新旧子节点列表
  • 操作类型
    • INSERT:新增节点
    • MOVE:移动已有节点
    • REMOVE:删除废旧节点

key的优化机制

场景无 Key有 Key
列表项顺序变化索引变化导致全部重新渲染识别移动项,仅调整 DOM 顺序
列表中间插入项后续项索引变化触发多节点更新仅插入新节点,后续项无变化
删除中间项后续项索引变化触发多节点更新仅删除目标节点

7.性能优化实践

1.key的使用原则
场景无 Key有 Key
列表项顺序变化索引变化导致全部重新渲染识别移动项,仅调整 DOM 顺序
列表中间插入项后续项索引变化触发多节点更新仅插入新节点,后续项无变化
删除中间项后续项索引变化触发多节点更新仅删除目标节点

7.性能优化实践

1.key的使用原则
  • 使用数据唯一的标识(如id)
  • 避免随机数或索引(不稳定)
2.避免跨层级移动节点
  • 修改父节点类型会导致子树重建
3.减少顶层节点类型变化
  • 保持稳定的组件结构
4.复杂列表优化
  • 虚拟滚动(如reac-window)
  • 分页加载

高频面试题

1.为什么列表必须使用key?
  • 帮助React识别元素身份,在顺序变化时高效复用DOM节点,避免以下问题:
    • 不必要的子组件重新渲染
    • 表单元素状态错乱(如输入框内容错位)
2.如何强制组件重新挂载?
  • 改变key值触发销毁/重建
 1<UserProfile key={user.id} user={user} />
3.Diff算法能完全避免DOM操作吗?
  • 不能。目的是最小化操作次数,但无法消除必要的更新(如数据变化必然导致DOM修改)

6.Fiber

1.Fiber的诞生背景

1.旧版协调算法瓶颈
  • 递归不可中断:同步遍历整个虚拟DOM树,长时间占用主线程
  • 卡顿问题:复杂组件树更新导致掉帧(如大型列表、动画场景)
2.目标
  • 实现增量渲染,支持异步可中断的更新

2.Fiber核心设计思想

1.时间切片
  • 将渲染任务拆分为多个小任务(Fiber节点)
  • 利用浏览器空闲时段(requestIdleCallback)分片执行
2.优先级调度
  • 用户交互(如输入)优先于数据更新(如API响应)
3.可恢复工作单元
  • 保存中间状态,允许暂停/恢复渲染流程

3.Fiber节点数据结构

每个Fiber节点对应一个组件实例或DOM节点,包含以下核心属性:

属性类型作用
typeString/Object组件类型(如 'div'、函数组件引用)
stateNodeObject对应的 DOM 节点或类组件实例
childFiber第一个子节点
siblingFiber下一个兄弟节点
returnFiber父节点
pendingPropsObject新传入的 props
memoizedPropsObject上一次渲染使用的 props
memoizedStateObject上一次渲染后的 state(如 Hooks 链表)
effectTagNumber标记副作用类型(如 PlacementUpdateDeletion
alternateFiber指向当前 Fiber 的镜像(用于 Diff 比较)

4.Fiber双缓冲机制

React维护两颗Fiber树已确保更新无冲突

  1. Current Tree:当前已渲染的UI对应的Fiber树
  2. WorkInProgress Tree:正在构建的新Fiber树
  • 切换流程:更新完成后,WorkInProgress Tree树变为Current树
 1初始渲染
 2Current Tree: null
 3WorkInProgress Tree: → 构建完成提交后成为 Current Tree
 4
 5更新阶段
 6Current Tree ←→ WorkInProgress Tree复用或新建节点

3.Fiber渲染

1.协调阶段
  • 目标:生成副作用列表,不修改DOM
  • 过程
    1. 递阶段:调用render生成子节点,标记变化
    2. 归阶段:向上回溯收集副作用
  • 可中断:根据剩余时间片暂停/恢复遍历
2.提交阶段
  • 目标:同步执行所有DOM变更
  • 过程
    1. Before Mutation:调用getSnapshotBeforeUpdate
    2. Mutation:执行DOM增删改
    3. Layout:调用useLayoutEffectcomponentEDidMount/Update
  • 不可中断:避免中间状态导致UI不一致

6.优先级调度模型

​优先级​​对应场景​
ImmediatePriority同步任务(如 flushSync
UserBlockingPriority用户交互(点击、输入)
NormalPriority数据更新、网络响应
LowPriority过渡更新(Concurrent 模式)
IdlePriority空闲时执行的任务
  • 调度策略:高优先级任务可中断低优先级任务

7.Fiber对生命周期的影响

1.废弃生命周期:
  • componentWillMount、componentWillReceiveProps、componentWillUpdate
  • 原因:异步可渲染可能导致多次调用,引发副作用错误
2.新增API
  • getDerivedStateFromProps(静态方法,替代componentWillReceiveProps
  • getSnapshotBeforeUpdate(替代componentWillUpdate

8.Fiber与并发模式

1.并发特性
  • useTranstion:标记非紧急更新,可被高优先级任务打断
  • Suspense:等待异步数据加载时显示回退UI
2.启用方式
 1ReactDOM.createRoot(document.getElementById('root')).render(<App />);

9.高频面试题

1.Fiber如何实现可中断更新?
  • 通过链表结构存储Fiber节点,记录遍历进度。每次处理一个Fiber后检查剩余时间片,不足时保存当前进度,将控制权交还浏览器
2.协调阶段和提交阶段的区别
  • 协调阶段负责计算变更(可中断),提交阶段执行DOM操作(同步不可中断)
3.为什么需要双缓冲机制?
  • 保证渲染过程中Current Tree始终完整可用,避免更新过程中出现UI不一致

二、状态管理

1.状态管理的核心

  1. 组件通信:跨层级组件共享数据
  2. 状态同步:多个组件依赖统一数据源
  3. 副作用管理:异步操作(API请求、定时器)与状态更新的协调
  4. 调试追踪:复杂应用中状态变更的可预测性

2.React内置解决方案

1.组件状态(useState/useReducer)

  • 适用场景:组件内部私有状态(如表单输入、UI切换)
 1const [count, setCount] = useState(0);
 2
 3
 4const reducer = (state, action) => {
 5  switch (action.type) {
 6    case 'increment': return { count: state.count + 1 };
 7    default: return state;
 8  }
 9};
10const [state, dispatch] = useReducer(reducer, { count: 0 });

2.Context API

  • 适用场景:中低频次更新的全局状态(如主题、用户身份)
 1const ThemeContext = createContext();
 2const UserContext = createContext();
 3
 4const UserPanel = memo(() => {
 5    const user = useContext(UserContext);
 6    return <div>{user.name}</div>;
 7})

3.主流状态管理库对比

​方案​​核心模式​​优点​​缺点​​适用场景​
​Redux​单一 Store + Flux严格的数据流、强大的中间件生态样板代码多、学习曲线陡峭大型复杂应用、需要时间旅行调试
​MobX​响应式编程代码简洁、自动追踪依赖黑盒机制、过度渲染风险中小型应用、快速开发
​Recoil​原子状态 + Selector细粒度控制、原生支持异步较新、生态不成熟需要局部状态优化的场景
​Zustand​轻量 StoreAPI 简单、无样板代码功能相对基础中小项目、替代部分 Redux 用例
​Context + Hooks​组合式状态零依赖、React 原生支持性能问题、缺乏中间件简单全局状态管理

4.Redux核心原理与最佳实践

1.三大原则:

  • 单一数据源:整个应用状态存储在一个Store中
  • 只读State:通过Action修改意图
  • 纯函数Reducer:接收旧State和Action,返回新State

2.现代Redux开发(Redux Toolkit)

 1const counterSlice = createSlice({
 2    name: 'counter',
 3    inintialState: 0,
 4    reducers: {
 5        increment: state => state + 1,
 6        decrement: state => state - 1
 7    }
 8}); 
 9
10const store = configureStore({
11    reducer: {
12        counter: counterSlice.reducer
13    }
14});
15
16const Counter = () => {
17    const count = useSelector(state => state.counter);
18    const dispach = useDispatch();
19    return <button onClick={() => dispatch(counterSlice.actions.increment())}{count}</button> 
20}

3.中间件应用

 1const fetchUser = () => async (dispatch) => {
 2    dispatch({ type: 'USER_REQUEST' });
 3    try {
 4        const res = await api.getUser();
 5        dispatch({ type: 'USER_SUCCESS', payload: res.data });
 6    } catch (err) {
 7        dispatch({ type: 'USER_FAILURE', error: err.message });
 8    }
 9}
10
11const logger = store => next => action => {
12    console.log('dispatching:', action);
13    let result = next(action);
14    console.log('next state:', store.getState());
15    return result;
16}

5.状态管理选型

1.评估应用规模

  • 小型应用:Context + useState/useReducer
  • 中大型应用:Redux/Mobx

2.团队熟悉度

  • 已有Redux经验:Redux Toolit
  • 偏好响应式:Mobx

3.性能需求

  • 高频更新:Recoil原子状态
  • 复杂异步:Redux + Saga/Thunk

4.开发效率

  • 快速迭代:Zustand/Jotai
  • 长期维护:Redux(强约束性)

6.高频面试题

1.redux如何避免不必要的重新渲染?

  • 使用reselect创建记忆化Selector,避免重复计算
  • 结合React.memo对组件进行浅比较
 1const selectUser = state => state.user;
 2const selectUserName = createSelector(
 3  [selectUser],
 4  (user) => user.name
 5);

2.Mobx的响应式原理

  • 通过ES6 ProxyObject.defineProperty追踪属性访问,自动建立观察者-被观察者关系,状态变更时触发依赖组件更新

3.如何解决Context API的性能问题?

  • 拆分多个Context隔离变化
  • 使用useMemo缓存Provider的value
 1const ThemeProvider = ({ children }) => {
 2    const [theme, setTheme] = useState('light');
 3    const value = useMemo(() => ({ theme, setTheme }), [theme]);
 4    return <ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>;
 5}

4.状态管理的不可变性为何重要?

  • 确保状态变更可预测,便于调试追踪
  • 支持React的浅比较优化策略(如shouldComponentUpdate
个人笔记记录 2021 ~ 2025