面试官:用代码简单实现一下useEffect的原理 - 我:这不有手就行

面试官:用代码简单实现一下useEffect的原理

React的useEffect钩子是React函数组件中处理副作用(例如API请求、订阅或手动修改DOM等)的重要工具。在本文中,将通过一个简单的例子解释如何用代码实现useEffect的基本原理。

首先,定义了两个全局变量,用于跟踪不同的副作用状态:

javascript 复制代码
let prevDepsAry = []; // 存放依赖项数组的上一次值
let effectIndex = 0; // 当前副作用索引

prevDepsAry用于记录上次渲染时的依赖项数组,effectIndex用来标示当前处理的副作用的位置。

接下来是useEffect函数的实现:

javascript 复制代码
function useEffect(callback, depsAry) {
  // 校验callback是否为函数
  if (Object.prototype.toString.call(callback) !== '[object Function]') {
    throw new Error('useEffect的第一个参数必须是一个函数');
  }

  // 不提供依赖项数组的情况下,默认每次渲染都执行callback
  if (typeof depsAry === 'undefined') {
    callback();
  } else {
    // 校验depsAry是否为数组
    if (Object.prototype.toString.call(depsAry) !== '[object Array]') {
      throw new Error('useEffect的第二个参数必须是数组');
    }

    // 获取依赖项数组的前一个值
    let prevDeps = prevDepsAry[effectIndex];

    // 比较依赖项数组的每一个值,确定是否发生变化
    let hasChanged = prevDeps ? !depsAry.every((dep, index) => dep === prevDeps[index]) : true;

    // 如果依赖项变化或者是首次渲染(hasChanged为true),则执行callback
    if (hasChanged) {
      callback();
    }

    // 将当前的依赖项数组存储起来,用于下次渲染时比较
    prevDepsAry[effectIndex] = depsAry;
  }

  // 增加副作用索引,准备下一个副作用
  effectIndex++;
}

useEffect的实现逻辑为:

  1. 验证传入的callback是否为函数。如果不是,抛出错误。
  2. 如果没有传入depsAry,那么每次组件渲染时都执行callback
  3. 如果传入了depsAry,首先验证其为数组。然后,获取上一次的依赖数组并与当前数组逐项比较。如果存在差异或者是首次渲染,则执行callback
  4. 更新prevDepsAry的对应项,在下次组件渲染时用作比较。
  5. effectIndex自增,确保下一个useEffect的处理使用正确的索引。

下一步定义了render函数:

javascript 复制代码
function render() {
  // 重设副作用索引
  effectIndex = 0;
  // 使用ReactDOM来渲染App组件
  ReactDOM.render(<App />, document.getElementById('root'));
}

在每次渲染前,effectIndex必须重置为0,这保证了useEffect在处理依赖项时的正确性。

最后,在App组件中使用useEffect

javascript 复制代码
function App() {
  // 使用自定义的useState和useEffect

  useEffect(() => {
    console.log('副作用函数执行了');
    // 在这里可能会执行如API请求、订阅事件等具有副作用的操作
    // 当依赖项发生变化时,这里的代码会被执行
  }, [/* 依赖项数组 */]);

  return (
    // 组件的JSX结构
    <div></div>
  );
}

在组件中通过调用useEffect,并传递一个执行副作用操作的函数以及依赖项数组,实现了依赖项变化时执行副作用逻辑的需求。

最后一步,调用render()以触发首次渲染。

通过上述代码,我单地回答了如何实现useEffect的问题。当然,实际的ReactuseEffect实现更加复杂,并且涉及到调度和清理操作,但上面的代码为理解其基本原理提供了良好的起点。

相关推荐
光头程序员7 小时前
grid 布局react组件可以循数据自定义渲染某个数据 ,或插入某些数据在某个索引下
javascript·react.js·ecmascript
limit for me7 小时前
react上增加错误边界 当存在错误时 不会显示白屏
前端·react.js·前端框架
浏览器爱好者7 小时前
如何构建一个简单的React应用?
前端·react.js·前端框架
VillanelleS11 小时前
React进阶之高阶组件HOC、react hooks、自定义hooks
前端·react.js·前端框架
傻小胖11 小时前
React 中hooks之useInsertionEffect用法总结
前端·javascript·react.js
flying robot21 小时前
React的响应式
前端·javascript·react.js
GISer_Jing1 天前
React+AntDesign实现类似Chatgpt交互界面
前端·javascript·react.js·前端框架
智界工具库1 天前
【探索前端技术之 React Three.js—— 简单的人脸动捕与 3D 模型表情同步应用】
前端·javascript·react.js
我是前端小学生1 天前
我们应该在什么场景下使用 useMemo 和 useCallback ?
react.js
我是前端小学生1 天前
讲讲 React.memo 和 JS 的 memorize 函数的区别
react.js