实现 render 函数 - 初始化更新队列

接下来,需要实现 render 函数,进行第一次挂载。

jsx 复制代码
const element = (
  <h1>
    hello <span>test</span> children
  </h1>
);

const root = createRoot(document.getElementById("root"));
root.render(element);

由之前的文章可知,root 是一个 ReactDOMRoot 实例。所以直接在 ReactDOMRoot 原型上添加 render 方法。

render 复制代码
// 注意 children 为传入的虚拟DOM 即 element
ReactDOMRoot.prototype.render = function (children) {
  // 这个 root 是 FiberRootNode 实例
  const root = this._internalRoot;
  // 更新容器
  updateContainer(children, root);
};

再回忆一下 HostRootFiberFiberRootNode

由于是初次渲染,所以第一次处理 HostRootFiber 的更新队列。

updateContainer 复制代码
/**
 * 更新容器 把虚拟DOM 变成真实DOM 插入容器
 * @param {*} element 虚拟DOM
 * @param {*} container 容器 FiberRootNode
 * @returns
 */
export function updateContainer(element, container) {
  // HostRootFiber 根 fiber
  const current = container.current;
  // 创建更新
  const update = createUpdate(eventTime, lane);
  // 需要更新的虚拟DOM
  update.payload = { element };
  // 添加更新
  // 返回根节点
  const root = enqueueUpdate(current, update);
  scheduleUpdateOnFiber(root);
}

更新队列的结构

initialUpdateQueue 可以看出,HostRootFiberupdateQueue 的结构为一个对象,对象中有个 shared 属性, shared 中有个 pending 属性指向循环链表。

updateQueue 复制代码
export function initialUpdateQueue(fiber) {
  const queue = {
    shared: {
      pending: null, // 循环链接,指向链表中最后一个 update
    },
  };
  fiber.updateQueue = queue;
}

由图可知,pending 永远指向循环链表中最后一个元素。由此可以实现 enqueueUpdate 函数。

enqueueUpdate 复制代码
export function enqueueUpdate(fiber, update) {
  const updateQueue = fiber.updateQueue;
  const sharedQueue = updateQueue.shared;
  const pending = sharedQueue.pending;
  if (pending === null) {
    // 第一次入队,指向自己
    update.next = update;
  } else {
    // 当队列不为空,新入的更新指向第一个更新
    update.next = pending.next;
    // pending 指向链表末尾
    pending.next = update;
  }
  // 指向最后一个 update
  sharedQueue.pending = update;
  // 找到根节点并返回
  return markUpdateLaneFromFiberToRoot(fiber);
}

执行 enqueueUpdate 后便生成了更新队列。

需要注意的是,当前更新中有一个 payload 参数,值为虚拟 DOM。

相关推荐
乘风gg20 分钟前
为什么AI 时代来临,大部分人吃不到红利
前端·ai编程·claude
恋猫de小郭41 分钟前
Android 限制侧载新进展,谷歌联合国内厂商推验证计划
android·前端·flutter
IT_陈寒1 小时前
Redis内存爆了,原来我漏掉了这个致命配置
前端·人工智能·后端
恋猫de小郭1 小时前
解读 Android 17 全新内存限制,有没有“豁免”后门?
android·前端·flutter
Hyyy2 小时前
理解LLM的基本工作原理:预训练、微调、推理的区别
前端
Gatlin3 小时前
前端逆向与反逆向:一场猫鼠游戏的底层逻辑与实战
前端
代码煮茶3 小时前
React 组件封装方法论 —— 以 Todo App 为例
javascript·react.js
Pedantic3 小时前
本地通知(Local Notifications)学习笔记
前端
森蓝情丶3 小时前
我给 AI 搭了个法庭:一个前端仔的 LangGraph 实战全记录
前端·后端
爱勇宝3 小时前
干了近 8 年,一夜之间被裁:AI 时代,程序员最该害怕的不是 AI
前端·后端·程序员