vue3源码解析 --- 组件渲染:vnode 到真实 DOM 是如何转变的

简介: vue3源码解析 --- 组件渲染:vnode 到真实 DOM 是如何转变的

vue3源码解析 --- 组件渲染:vnode 到真实 DOM 是如何转变的


源码解析略复杂,这边先实现超级简单版,缕清思路,然后在后文展开细节,水平有限,如有不对,欢迎讨论~

基础知识:vnode是啥

看到vnode,就想到这是一个对象就行了,主要是描述普通HTML标签或者组件标签的,长这样

描述普通节点:

// <button class="btn" title="btn">click me</button>
const vnode = {
  type: 'button',
  props: { 
    'class': 'btn',
    title: 'btn'
  },
  children: 'click me'
}

描述组件节点:

// <custom-component msg="test"></custom-component>
const CustomComponent = {
  template:`<div>我是组件</div>`
}
const vnode = {
  // 注意此处的type是 对象
  type: CustomComponent,
  props: { 
    msg: 'test'
  }
}

基础知识:渲染vnode是啥

其实就是将{ type: 'button', children: 'click me' }转化成真实的dom节点,然后插入到文档中,文档自然就会去渲染了

function patch(vnode,container){
  // 将vnode转化成真实的元素
  const el = document.createElement(vnode.type)
  el.textContent = vnode.children
  // 将元素挂载到文档中,让文档去渲染展示
  container.appendChild(el)
}

特别简化版的vue的例子

此demo是了解组件渲染:vnode 到真实 DOM 是如何转变的

为了极致方便,只使用静态文件

下面可以在你的本地试试,两个文件:index.htmlvue.simple.js

<body>
  <div id="app">
  </div>
  <script src="vue.simple.js"></script>
  <script>
    const c1 = {
      name:'c1',
      template: `我是c1`
    }
    const App = {
      name:'app',
      components: {c1},
      // 组件里有组件的话
      template: `hello,<c1></c1>`
    }
    Vue.createApp(App).mount('#app')
  </script>
</body>
// vue.simple.js
const Vue = {
  createApp(rootComponent) {
    const app = {
      _component: rootComponent,
      mount(rootContainer) {
        // 这边是简化了,只写了组件节点的vnode写法
        const createVNode = rootComponent => ({ type: rootComponent, shapeFlag: 4 })
        // 创建根组件的 vnode
        const vnode = createVNode(rootComponent)
        // 利用渲染器渲染 vnode
        render(vnode, rootContainer)
      }
    }
    return app
  }
}
function render(vnode, container) {
  // 简单理解为就是执行patch,就是把vnode变成真实的dom,挂载到container上
  patch(vnode, container);
  function patch(vnode, container) {
    // 得到template的 children
    const getChildren = (vnode) => {
      const div = document.createElement('div')
      div.innerHTML = vnode.type.template
      return [...div.childNodes]
    }
    const children = getChildren(vnode)
    console.log(children)
    const frg = document.createDocumentFragment()
    children.forEach(item => {
      // 组件 这里为了方便 写死组件名了
      if (item.nodeType === 1 && item.nodeName === 'C1') {
        // 其实这里本应是patch,但放了方便简化了,大致就是组件内部的先挂载到组件标签的位置。组件是深度优先挂载。
        frg.appendChild(document.createTextNode(vnode.type.components.c1.template))
      }
      // 文本节点
      if (item.nodeType === 3) {
        frg.appendChild(document.createTextNode(item.textContent))
      }
    })
    // 子节点都创建完了之后,一起插入到文档中,从而渲染出来
    document.querySelector(container).appendChild(frg)
  }
}

效果图,超级简单:

网络异常,图片无法展示
|

下面分析源码,但真实的源码很复杂,我择其要点分析,原文01 | 组件渲染:vnode 到真实 DOM 是如何转变的?

应用程序初始化:createApp

其实index.html里核心代码就是Vue.createApp(App).mount('#app')

createApp就是创建了一个app对象

const createApp = (rootComponent) => {
  // 创建 app 对象
  const app = {
      _component: rootComponent,
      mount(rootContainer) {
        // 创建根组件的 vnode,例子里为了简单直接写死了 其实就是 {type:x, shapeFlag:x}
        const vnode = createVNode(rootComponent)
        // 利用渲染器渲染 vnode
        render(vnode, rootContainer)
      }
    }
  return app
}

核心渲染流程:创建 vnode 和渲染 vnode

创建vnode主要是createVNode

//  主要就是返回 { type, props, shapeFlag }
 function createVNode(type, props = null ,children = null) {
  if (props) { 
    // 处理 props 相关逻辑,标准化 class 和 style 
  }
  // 对 vnode 类型信息编码
  //  1就是普通标签,4就是组件标签 0是文本 其他类型暂不考虑
 const shapeFlag = isString(type) ? 1 : isObject(type)? 4 : 0
 const vnode = { type, props, shapeFlag, }
 // 标准化子节点,把不同数据类型的 children 转成数组或者文本类型
  normalizeChildren(vnode, children)
  return vnode
}

渲染 vnode主要是render

render(vnode, rootContainer)
const render = (vnode, container) => {
  // vnode是空的话有个销毁逻辑
  // 创建或者更新组件,最核心是patch,patch其实就根据组件、元素有对应的处理
  patch(container._vnode || null, vnode, container)
  // 缓存 vnode 节点,表示已经渲染
  container._vnode = vnode
}

原文的描述更多,最核心的点就是,patch的时候,是深度优先挂载(或更新)。

01 | 组件渲染:vnode 到真实 DOM 是如何转变的?

目录
相关文章
|
7月前
|
机器学习/深度学习 数据采集 JavaScript
用深度学习提升DOM解析——自动提取页面关键区块
本文介绍了一次二手车数据爬虫事故的解决过程,从传统XPath方案失效到结合深度学习语义提取的成功实践。面对懂车帝平台的前端异步渲染和复杂DOM结构,通过Playwright动态渲染、代理IP隐藏身份,以及BERT模型对HTML块级语义识别,实现了稳定高效的字段提取。此方法抗结构变化能力强,适用于复杂网页数据采集,如二手车、新闻等领域。架构演进从静态爬虫到动态爬虫再到语义解析,显著提升效率与稳定性。
277 13
用深度学习提升DOM解析——自动提取页面关键区块
|
9月前
|
资源调度 JavaScript
Vue 3 中如何通过状态管理库来更新虚拟 DOM?
Vue 3 中如何通过状态管理库来更新虚拟 DOM?
283 57
|
9月前
|
JavaScript 开发者
在 Vue 3 中,除了 `markRaw` 和 `reactive` 结合,还有其他方式手动更新虚拟 DOM 吗?
在 Vue 3 中,除了 `markRaw` 和 `reactive` 结合,还有其他方式手动更新虚拟 DOM 吗?
211 57
|
9月前
|
算法 测试技术 C语言
深入理解HTTP/2:nghttp2库源码解析及客户端实现示例
通过解析nghttp2库的源码和实现一个简单的HTTP/2客户端示例,本文详细介绍了HTTP/2的关键特性和nghttp2的核心实现。了解这些内容可以帮助开发者更好地理解HTTP/2协议,提高Web应用的性能和用户体验。对于实际开发中的应用,可以根据需要进一步优化和扩展代码,以满足具体需求。
906 29
|
9月前
|
数据采集 前端开发 JavaScript
金融数据分析:解析JavaScript渲染的隐藏表格
本文详解了如何使用Python与Selenium结合代理IP技术,从金融网站(如东方财富网)抓取由JavaScript渲染的隐藏表格数据。内容涵盖环境搭建、代理配置、模拟用户行为、数据解析与分析等关键步骤。通过设置Cookie和User-Agent,突破反爬机制;借助Selenium等待页面渲染,精准定位动态数据。同时,提供了常见错误解决方案及延伸练习,帮助读者掌握金融数据采集的核心技能,为投资决策提供支持。注意规避动态加载、代理验证及元素定位等潜在陷阱,确保数据抓取高效稳定。
290 17
|
9月前
|
前端开发 数据安全/隐私保护 CDN
二次元聚合短视频解析去水印系统源码
二次元聚合短视频解析去水印系统源码
383 4
|
8月前
|
机器学习/深度学习 数据采集 存储
深度学习在DOM解析中的应用:自动识别页面关键内容区块
本文探讨了如何通过深度学习模型优化东方财富吧财经新闻爬虫的性能。针对网络请求、DOM解析与模型推理等瓶颈,采用代理复用、批量推理、多线程并发及模型量化等策略,将单页耗时从5秒优化至2秒,提升60%以上。代码示例涵盖代理配置、TFLite模型加载、批量预测及多线程抓取,确保高效稳定运行,为大规模数据采集提供参考。
211 0
|
9月前
|
JavaScript 算法 前端开发
JS数组操作方法全景图,全网最全构建完整知识网络!js数组操作方法全集(实现筛选转换、随机排序洗牌算法、复杂数据处理统计等情景详解,附大量源码和易错点解析)
这些方法提供了对数组的全面操作,包括搜索、遍历、转换和聚合等。通过分为原地操作方法、非原地操作方法和其他方法便于您理解和记忆,并熟悉他们各自的使用方法与使用范围。详细的案例与进阶使用,方便您理解数组操作的底层原理。链式调用的几个案例,让您玩转数组操作。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
9月前
|
移动开发 前端开发 JavaScript
从入门到精通:H5游戏源码开发技术全解析与未来趋势洞察
H5游戏凭借其跨平台、易传播和开发成本低的优势,近年来发展迅猛。接下来,让我们深入了解 H5 游戏源码开发的技术教程以及未来的发展趋势。
|
9月前
|
存储 前端开发 JavaScript
在线教育网课系统源码开发指南:功能设计与技术实现深度解析
在线教育网课系统是近年来发展迅猛的教育形式的核心载体,具备用户管理、课程管理、教学互动、学习评估等功能。本文从功能和技术两方面解析其源码开发,涵盖前端(HTML5、CSS3、JavaScript等)、后端(Java、Python等)、流媒体及云计算技术,并强调安全性、稳定性和用户体验的重要性。

推荐镜像

更多
  • DNS