从头开始设计虚拟 DOM:分步指南

从头开始设计虚拟 dom:分步指南

如果您听说过 react 或 vue 等前端库,您可能遇到过术语 虚拟 dom。虚拟 dom 是一个聪明的概念,它可以通过提高 dom 更新效率来帮助加快 web 开发速度。

在本指南中,我们将详细介绍如何使用通用的类似代码的步骤从头开始实现简单的虚拟 dom。

什么是虚拟 dom?

虚拟 dom 只是真实 dom(网页结构)的轻量级内存表示。我们不是直接更新真实 dom(这很慢),而是首先对虚拟 dom 进行更改,弄清楚发生了什么变化,然后只更新真实 dom 中需要更新的部分。这可以节省时间并使您的应用程序运行得更快!

第 1 步:将虚拟 dom 表示为树

将网页的结构想象成一棵树,其中每个元素(如

)都是树中的一个“节点”。 虚拟 dom 节点 是一个代表这些元素之一的小对象。

这是一个例子:

virtual dom node:{  type: 'div',  props: { id: 'container' },  // attributes like id, class, etc.  children: [                 // children inside this element    {      type: 'p',              // a 

tag (paragraph) props: {}, children: ['hello, world!'] // text inside the

tag } ]}

这描述了一个 id=”container” 的

元素,其中包含一个带有文本 “hello, world!”. 的

元素。

要点:

每个节点都有一个类型(例如 div、p)。它可以有 props(如 id、class 等)。它还有可以是其他元素或文本的子元素。

第二步:将虚拟 dom 渲染到真实 dom

现在我们有了虚拟 dom,我们需要一种方法将其转换为页面上的真实 html。

让我们编写一个名为 render 的函数,它接收虚拟 dom 节点并将其转换为实际的 html 元素。

function render(vnode) {  // 1. create a real element based on the virtual dom type (e.g., div, p).  const element = document.createelement(vnode.type);  // 2. apply any attributes (props) like id, class, etc.  for (const [key, value] of object.entries(vnode.props)) {    element.setattribute(key, value);  }  // 3. process the children of this virtual dom node.  vnode.children.foreach(child => {    if (typeof child === 'string') {      // if the child is just text, create a text node.      element.appendchild(document.createtextnode(child));    } else {      // if the child is another virtual dom node, recursively render it.      element.appendchild(render(child));    }  });  return element;  // return the real dom element.}

这里发生了什么?

我们创建一个元素 (document.createelement(vnode.type))。我们添加任何属性(如 id、class 等)。我们通过添加文本或在每个子元素上再次调用渲染来处理子元素(文本或其他元素)。

第 3 步:比较(差异)新旧虚拟 dom

当我们的网络应用程序发生某些变化(例如文本或元素的样式)时,我们会创建一个新的虚拟 dom。但在更新真实 dom 之前,我们需要比较旧 virtual dom新 virtual dom 来找出发生了什么变化。这称为“比较”

让我们创建一个比较两个虚拟 dom 的函数:

function diff(oldvnode, newvnode) {  // if the node type has changed (e.g., from 'div' to 'p'), replace the node.  if (oldvnode.type !== newvnode.type) {    return { type: 'replace', newvnode };  }  // if the text inside has changed, we need to update the text.  if (typeof oldvnode === 'string' && oldvnode !== newvnode) {    return { type: 'text', newvnode };  }  // compare the properties (like id, class, etc.).  const proppatches = diffprops(oldvnode.props, newvnode.props);  // compare the children.  const childpatches = diffchildren(oldvnode.children, newvnode.children);  return { type: 'update', proppatches, childpatches };}

差异如何工作:

节点类型更改:如果元素类型发生更改(例如将

更改为

),我们将其标记为替换。

文本更改:如果内部文本发生更改,我们会更新文本。属性和子元素:我们检查元素的任何属性(props)或子元素是否已更改。

第 4 步:修补真实 dom

一旦我们知道发生了什么变化,我们就需要将这些更改应用到真实的 dom 上。我们将此过程称为修补

修补功能的外观如下:

function patch(parent, patches, index = 0) {  if (!patches) return;  // If there are no changes, do nothing.  const element = parent.childNodes[index];  // The element we’re patching.  switch (patches.type) {    case 'REPLACE':      // Replace the old element with a new one.      parent.replaceChild(render(patches.newVNode), element);      break;    case 'TEXT':      // Update the text content of the element.      element.textContent = patches.newVNode;      break;    case 'UPDATE':      // Update the element’s properties and children.      patchProps(element, patches.propPatches);      patches.childPatches.forEach((childPatch, i) => patch(element, childPatch, i));      break;  }}

关键操作:

替换:将一个元素完全替换为另一个元素。文本:更新元素内的文本。更新:根据更改更新属性和子项。

虚拟 dom 流程总结:

虚拟 dom 树: 我们创建一个树状结构,表示网页上的元素及其层次结构。渲染为真实 dom: 此虚拟 dom 被转换为真实的 html 元素并放置在页面上。比较算法:当发生变化时,我们将旧的 virtual dom 与新的进行比较以找出差异。修补真实 dom: 以最有效的方式将这些更改应用到真实 dom。

最后的想法

虚拟 dom 是一个强大的工具,通过减少对真实 dom 的不必要的更改,可以更快地更新用户界面。通过实现虚拟 dom,我们可以优化 web 应用程序更新和渲染元素的方式,从而带来更快、更流畅的用户体验。

这是虚拟 dom 概念的基本实现,但您现在已经有了理解 react 等框架如何使用它的基础!

以上就是从头开始设计虚拟 DOM:分步指南的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1493484.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
在选择元素中动态创建选项
上一篇 2025年12月19日 15:43:32
如何在任何 Nextjs 应用程序中处理 Cookie 同意
下一篇 2025年12月19日 15:43:48

相关推荐

  • Golang空接口如何应用在项目中

    空接口可用于接收任意类型值,常见于日志函数、通用数据结构、JSON动态解析及配置驱动逻辑,提升代码灵活性,但需配合类型断言确保安全,避免滥用以降低维护成本。 空接口 interface{} 在 Go 语言中是一个非常灵活的类型,它可以存储任何类型的值。虽然它牺牲了一部分类型安全,但在实际项目中合理使…

    2026年5月10日
    100
  • Go语言接口与切片:如何识别和操作[]interface{}

    本文将深入探讨Go语言中如何识别和操作`[]interface{}`类型的切片。我们将介绍类型断言(Type Assertion)的关键作用,并通过`switch`语句演示如何安全地检测`[]interface{}`类型,并进而遍历其内部元素。文章旨在提供清晰的示例代码和专业指导,帮助开发者有效地处…

    2026年5月10日
    000
  • JavaScript计算器开发:解决数值显示与初始化问题

    本教程深入探讨了使用JavaScript构建计算器时常见的数值显示异常问题,特别是由于类属性未初始化导致的`Cannot read properties of undefined`错误。我们将详细分析问题根源,并通过在构造函数中调用初始化方法来解决该问题,同时优化显示逻辑,确保计算器功能稳定且界面显…

    2026年5月10日
    000
  • Go语言与Microsoft SharePoint集成指南

    Go语言可以有效集成Microsoft SharePoint,主要通过两种途径:一是利用SharePoint提供的RESTful API进行数据交互,Go的标准HTTP客户端库即可轻松实现;二是通过SharePoint应用模型开发自托管应用,这种模型支持使用包括Go在内的任何语言编写后端逻辑。 1.…

    2026年5月10日
    000
  • javascript生命周期钩子是什么_组件有哪些关键阶段?

    JavaScript原生无生命周期钩子,这是Vue、React等框架为组件设计的机制;Vue按创建、挂载、更新、卸载四阶段提供对应钩子,React类组件有明确生命周期方法,函数组件则通过useEffect模拟,其核心价值在于精准控制执行时机以避免DOM操作错误和内存泄漏。 JavaScript 本身…

    2026年5月10日
    100
  • JavaScript动态下拉菜单:实现日期选项与价格计算关联

    在现代web应用中,动态生成表单元素并使其具备交互逻辑是常见的需求。特别是在需要根据用户选择调整价格或服务参数的场景下,下拉菜单()常被用来展示一系列选项。本教程将指导您如何利用javascript动态生成一个包含日期选项的下拉菜单,并为每个选项关联一个具体的数值(如剩余天数),进而实现一个基于用户…

    2026年5月10日
    000
  • 前后端分离项目中,如何解决“net::ERR_CONNECTION_REFUSED”错误?

    修复“bug:net::err_connection_refused”后端代码中的错误 在开发前后端分离项目时,使用vue2前端和fastapi后端,前端希望通过“http://10.96.67.161:8081/uploadimg/”接口传输图片给后端,但遇到了“post http://10.96…

    2026年5月10日
    000
  • 让我们只用一根安装线就可以使网络响应起来吗?我正在寻找贡献者!

    最近我发布了一个 npm 包,其使命如标题所示:让项目只需一行代码即可响应! 我与您分享响应式应用程序 [beta] 包 我花了几年时间尝试和开发这项技术,目前包括: 动态设置 html 标签字体大小(通过 js 脚本),考虑:(1) 屏幕分辨率和 (2) 浏览器字体大小(用于网络可访问性)将像素定…

    2026年5月10日
    000
  • Go语言中实现策略模式:灵活处理多源数据与格式转换

    本文探讨了如何在go语言中实现策略模式,以优雅地处理多源数据收集与多格式数据转换的场景。通过定义清晰的接口和具体的策略实现,结合go语言简洁的特性,展示了两种将策略集成到工作流中的方法,强调了go中接口驱动的灵活性。 在软件开发中,我们经常面临需要处理多种算法或行为,并根据具体情况选择其中之一的场景…

    2026年5月10日
    000
  • 如何在H5环境中实现EXCEL和Word文件的在线预览?

    H5环境下EXCEL和Word文件在线预览技术详解 在如今的Web应用中,在线预览文档功能至关重要,尤其对于需要处理大量文件的企业应用。本文将介绍两种无需插件,直接在浏览器中预览EXCEL和Word文件的方法。 企业对在线文档预览的需求日益增长,以提高效率和协同性。下面我们探讨两种有效的H5在线预览…

    2026年5月10日
    000
  • JS如何实现策略模式

    策略模式通过封装算法使其可互换,JavaScript中利用函数作为一等公民实现,适用于表单验证等场景,结合工厂模式提升灵活性,但应避免过度设计。 策略模式的核心在于定义一系列算法,并将每一个算法封装起来,使它们可以相互替换。这使得算法可以在不影响客户端的情况下发生变化。在JS中,这可以通过函数作为一…

    2026年5月10日
    000
  • 小程序如何自动切换语言,才能兼顾精准性和适用性?

    根据小程序实现自动切换语言 对于多语言环境的小程序,如何自动切换语言是一个常见问题。 方法一:根据用户定位 这种方法基于用户定位来获取所在国家,并根据国家对应语言设置。然而,这种方法存在一定的局限性,如用户可能不会允许位置共享。 方法二:通过 wx.getsysteminfo 这是微信官方提供的 a…

    2026年5月10日
    000
  • WebView2 无法接收打包 Vue 项目数据:如何解决 C# 与 Vue 项目通信问题?

    webview2 无法接收打包 vue 项目发送的数据 在使用 webview2 嵌入 vue 项目时,开发者可能会遇到无法接收 c# 传输的数据的问题。要解决此问题,首先需要检查 webview2 控件加载的 html 页面的正确性。 一般来说,如果在加载独立的 html 页面(例如 test.h…

    2026年5月10日
    000
  • 优化React-Redux应用中的用户与受保护数据按需加载

    本教程旨在解决React-Redux应用中用户数据和受保护API密钥在用户未登录时仍被请求,导致401错误的问题。通过引入条件性Redux状态初始化和动作分发逻辑,确保只有在用户被认为已认证时才发起相关的API请求,从而优化应用性能,减少不必要的网络流量和控制台错误。 在构建现代Web应用时,尤其是…

    2026年5月10日
    000
  • Go语言并发二叉树遍历:通道关闭与等价性判断的优雅方案

    本文探讨了在Go语言中并发遍历二叉树时,如何正确处理通道(channel)的关闭时机问题,尤其是在递归函数中。通过结合defer语句和闭包(closure)的巧妙运用,提供了一种优雅且健壮的解决方案,确保通道在所有值发送完毕后才被关闭,进而实现两个二叉树的等价性判断。 1. 并发遍历二叉树的需求与挑…

    2026年5月10日
    000
  • Vue.js前端生成带分页符的Word文档:挑战与解决方案? 或 如何在Vue.js前端生成包含分页符的Word文档?

    vue.js前端生成word文档并插入分页符的难题 许多开发者希望在Vue.js前端直接将HTML转换为包含分页符的Word文档。本文分析实现此目标的方法以及面临的挑战。 用户尝试使用page-break-after: always属性(或其替代属性break-after)在生成的Word文档中添加…

    2026年5月10日
    000
  • Golang环境变量配置自动化脚本方法

    答案:通过编写Shell脚本自动化配置Go环境变量,可实现GOROOT、GOPATH、GOBIN及PATH等变量的自动设置,提升开发效率。具体做法是创建setup_go_env.sh脚本,定义GOROOT为Go安装路径(如/usr/local/go),GOPATH为工作区(如~/go_project…

    2026年5月10日
    100
  • 前端实现记住密码与自动填充_javascript技巧

    正确使用表单标签与属性、支持“记住我”功能、避免破坏自动填充机制、测试浏览器兼容性可实现稳定自动填充。1. 使用标准input类型并设置autocomplete属性为username和current-password;2. 登录成功后通过localStorage保存用户名,页面加载时恢复;3. 避免…

    2026年5月10日
    000
  • 深入理解Go语言中的短声明:=与长声明var

    Go语言提供了两种主要的变量声明和初始化方式:短声明:=和长声明var。:=主要用于函数内部,实现变量的声明与初始化,并常用于控制流语句中以限制变量作用域,例如在if语句中处理错误。而var则更为通用,可用于包级别或函数内部,支持显式类型声明、不带初始化的声明以及批量声明,提供了更大的灵活性。 1.…

    2026年5月10日
    000
  • JavaScript中动态生成HTML链接:正确使用模板字面量嵌入URL

    本文深入探讨了在javascript中动态生成html链接时,如何正确地将变量(尤其是url)嵌入到`href`属性中。通过分析常见的错误,即混淆javascript的模板字面量与框架特有的模板语法,文章详细演示了使用es6模板字面量`${}`进行字符串插值的正确方法,确保动态链接能够被浏览器正确解…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信