React 使用 MessageChannel 实现异步更新

React 通过异步合并更新,提高渲染速度。

RequestIdleCallback 的不足

  • 兼容性差-RequestIdleCallback 并不是对所有浏览器都支持的。
  • 工作帧率低,只有 20FPS,

SetTimeout 是最后的兜底

setTimeout 精确度不够,即使你把延迟时间设置成 0,实际上还会有 4ms 的延迟时间,只能作最后的兜底。

js 复制代码
setTimeout(() => {
    // 执行异步操作
}, 0);

如何使用 MessageChannel?

入口函数 requestHostCallback:

js 复制代码
// 是否有其他异步任务在运行(异步任务指的是 React 创建的异步任务)
// 只有一个主线程,每次只能执行一个任务
// isMessageLoopRunning 类似一把锁,没有任务执行时是 false
// 当要执行一个任务 A 时,isMessageLoopRunning 被设置为 true,防止其他任务执行
// 执行完任务 A,再把 isMessageLoopRunning 设置为 false,才可继续执行任务 B。
let isMessageLoopRunning = false;

function requestHostCallback() {
  if (!isMessageLoopRunning) {
    isMessageLoopRunning = true;
    // 执行 work
    schedulePerformWorkUntilDeadline();
  }
}

使用 MessageChanne 创建宏任务,实现异步任务队列,以实现异步更新,确保 React 在执行更新时能够合并多个更新操作,并在下一个宏任务中一次性更新,以提高性能并减少不必要的重复渲染,从而提高页面性能和用户体验。

js 复制代码
const channel = new MessageChannel();
const port = channel.port2;
// 监听
channel.port1.onmessage = performWorkUntilDeadline;
// 在一个时间切片内执行,直到时间切片结束
function schedulePerformWorkUntilDeadline() {
  // 执行 work,即发送消息
  port.postMessage(null);
}

function performWorkUntilDeadline() {
  if (isMessageLoopRunning) {
    const currentTime = getCurrentTime();
    // 记录了一个 work 的起始时间,其实就是一个时间切片的起始时间,这是个时间戳
    // 是否结束通过 shouldYieldToHost 函数判断
    startTime = currentTime;
    let hasMoreWork = true;
    try {
      hasMoreWork = flushWork(currentTime);
    } finally {
      if (hasMoreWork) {
        // 有更多的任务
        schedulePerformWorkUntilDeadline();
      } else {
        // 没有任务,这轮 work 结束
        isMessageLoopRunning = false;
      }
    }
  }
}

function flushWork(initialTime: number) {
  // 结束主线程调度
  isHostCallbackScheduled = false;
  // 开始执行任务
  isPerformingWork = true;

  try {
    return workLoop(initialTime);
  } finally {
    currentTask = null;
    // 结束执行任务
    isPerformingWork = false;
  }
}

其中 workLoop 函数和 shouldYieldToHos 函数参考文章 认识 React 中的时间切片

相关推荐
lbb 小魔仙12 小时前
【HarmonyOS实战】React Native 表单实战:自定义 useReactHookForm 高性能验证
javascript·react native·react.js
早點睡39016 小时前
高级进阶 ReactNative for Harmony 项目鸿蒙化三方库集成实战:react-native-drag-sort
react native·react.js·harmonyos
C澒17 小时前
Vue 项目渐进式迁移 React:组件库接入与跨框架协同技术方案
前端·vue.js·react.js·架构·系统架构
发现一只大呆瓜18 小时前
虚拟列表:从定高到动态高度的 Vue 3 & React 满分实现
前端·vue.js·react.js
全栈探索者19 小时前
列表渲染不用 map,用 ForEach!—— React 开发者的鸿蒙入门指南(第 4 期)
react.js·harmonyos·arkts·foreach·列表渲染
程序员Agions20 小时前
useMemo、useCallback、React.memo,可能真的要删了
前端·react.js
NEXT0620 小时前
React Hooks 进阶:useState与useEffect的深度理解
前端·javascript·react.js
早點睡3901 天前
基础入门 React Native 鸿蒙跨平台开发:react-native-flash-message 消息提示三方库适配
react native·react.js·harmonyos
早點睡3901 天前
高级进阶 ReactNative for Harmony项目鸿蒙化三方库集成实战:react-native-image-picker(打开手机相册)
react native·react.js·harmonyos
早點睡3901 天前
基础入门 React Native 鸿蒙跨平台开发:react-native-easy-toast三方库适配
react native·react.js·harmonyos