Vue渲染器解析

渲染器是 Vue 与浏览器之间的「翻译官」。它拿到一份用 JavaScript 对象描述的 UI(虚拟 DOM),然后精准地创建、更新、销毁真实 DOM,同时把响应式数据和渲染函数绑定成一条自动刷新的流水线。

一、核心职责:挂载 + 更新 + 卸载

渲染器的使命只有三句话:

  • 首次出现时把虚拟节点挂载成真实节点;
  • 数据变化时用最小代价更新节点;
  • 节点消失时把 DOM 和副作用清理干净。

所有细节都围绕这三件事展开。

二、从代码看挂载全流程

js 复制代码
function mountElement(vnode, container) {
  // 1. 创建元素
  const el = document.createElement(vnode.type);

  // 2. 处理属性
  if (vnode.props) {
    for (const key in vnode.props) {
      const value = vnode.props[key];
      if (key.startsWith('on')) {
        // 事件
        el.addEventListener(key.slice(2).toLowerCase(), value);
      } else if (key === 'class') {
        // class 归一化后一次性赋值
        el.className = normalizeClass(value);
      } else {
        // 普通属性
        el[key] = value;
      }
    }
  }

  // 3. 处理子节点
  if (typeof vnode.children === 'string') {
    el.textContent = vnode.children;
  } else if (Array.isArray(vnode.children)) {
    vnode.children.forEach(child => mountElement(child, el));
  }

  // 4. 插入文档
  container.appendChild(el);
}

示例代码已经覆盖「元素创建、属性绑定、事件监听、子节点递归、DOM 插入」五大动作。真实 Vue 只是在此基础上加了 patchFlag、Block Tree 等优化,逻辑完全一致。

三、元素挂载处理细节问题

1. 属性到底用 setAttribute 还是 elkey

普通字符串属性推荐 el[key],少一次字符串解析,性能更优。布尔属性如 disabled 必须 el.disabled = false,否则 setAttribute('disabled', 'false') 会把按钮禁用。只读属性如 form 只能 setAttribute,因为 el.form 是只读。

Vue 内部用 shouldSetAsProps 函数做决策:

js 复制代码
function shouldSetAsProps(el, key) {
  if (key === 'form' && el.tagName === 'INPUT') return false;
  return key in el;
}

先判断是否有对应 DOM 属性,再决定走哪条路,确保正确性与性能兼得。

2.class 的特殊处理:字符串、对象、数组一网打尽

模板中的 :class 可能是字符串、对象或数组,渲染器先用 normalizeClass 统一成空格分隔的字符串,再一次性赋给 el.className,避免多次 DOM 操作。

js 复制代码
function isString(value) {
  return typeof value === "string";
}

function isArray(value) {
  return Array.isArray(value);
}

function isObject(value) {
  return value !== null && typeof value === "object";
}

function normalizeClass(value) {
  let res = "";
  if (isString(value)) {
    res = value;
  } else if (isArray(value)) {
    // 如果是数组,递归调用 normalizeClass
    for (let i = 0; i < value.length; i++) {
      const normalized = normalizeClass(value[i]);
      if (normalized) {
        res += (res ? " " : "") + normalized;
      }
    }
  } else if (isObject(value)) {
    // 如果是对象,则检查每个 key 是否为真值
    for (const name in value) {
      if (value[name]) {
        res += (res ? " " : "") + name;
      }
    }
  }
  return res;
}
normalizeClass(['foo', { bar: true, baz: false }]) // → 'foo bar'

3.子节点的挂载

子节点可能是文本、数组或自定义组件。

  • 文本直接 textContent
  • 数组递归 mountElement
  • 组件则执行 mountComponent,组件再返回新的虚拟节点,继续递归。

整个页面就是一颗虚拟 DOM 树 深度优先地展开成真实 DOM 树。

js 复制代码
function mountElement(vnode, container) {
  const el = createElement(vnode.type);
  
  // 针对子节点进行处理
  if (typeof vnode.children === "string") {
    // 如果 children 是字符串,则直接将字符串插入到元素中
    setElementText(el, vnode.children);
  } else if (Array.isArray(vnode.children)) {
    // 如果 children 是数组,则遍历每一个子节点,并调用 patch 函数挂载它们
    vnode.children.forEach((child) => {
      patch(null, child, el);
    });
  }
  insert(el, container);
}
相关推荐
远航_14 分钟前
OpenSpec 完整详细介绍
前端·后端
召钱熏25 分钟前
状态枚举正确≠渲染正确:一个语音按钮的状态机边界修复实录
android·前端
Randyliu25 分钟前
20260508-Agent搭建记录以及对ReAct框架的理解
面试·agent
SkyWalking中文站26 分钟前
认识 Horizon UI · 1/17:SkyWalking 新一代可观测性控制台
运维·前端·监控
cidy_9826 分钟前
Dify 操作教程:工作流编排 & Chat 对话编排
前端·工作流引擎
tangdou36909865529 分钟前
AI真好玩系列-2分钟快速了解DeepAgents | Quick Guide to DeepAgents in 2 Minutes
前端·javascript·后端
小四的小六31 分钟前
AI Agent效果评测实战——搭完Agent才是噩梦的开始
前端
梨子同志39 分钟前
JavaScript
前端
彭于晏爱编程40 分钟前
纯 JS + Node,一个下午手搓了能读懂公司代码的 AI 助手,老板以为我转行了
前端·javascript
ZzT1 小时前
公司用 AI 筛简历,他写了个 AI 帮你挑公司
面试·aigc·ai编程