diff算法理解第一篇

在vue和react中,diff算法都是重要的部分,经历过jquery的前端开发同行都有体会,发请求拿到接口后,生成一个新的符合html的字符串,各种拼接,然后获取div的节点,在拼到获取的div的里面,很麻烦的,但是在diff里,就变得特别简单了,一个循环,可以搞定原来jquery的很多代码,下面,我说一下我对diff的理解。

所谓的diff算法,是dom的数据化,先处理数据,只操作一次dom,把由原来的拼接一个新的符合html字符串,在获取的div的节点里,塞入拼接好的字符串,在这里可以想一下,假如一个后台管理系统,在jquery的时代,页面进入展示表格,重新渲染一次,你点击查询按钮,如果和进入页面的查询条件没有变(假设返回数据没有变化),表格的内容是要在渲染一次的,但是在diff中,表格的dom是没有变化的,这就能看出性能的差距的。 举个例子

js 复制代码
    let box = document.getElementById("box")
    console.time("a")
    for(let i=0;i<=1000000;i++){
        box.innerHTML=i
    }
    console.timeEnd("a")

    console.time("b")
    let num =0
    for(let i=0;i<=1000000;i++){
        num=i
    }
    console.log(num)
    console.timeEnd("b")

运行结果

同样是循环1000000次,循环一次,操作dom一次,循环一次,操作dom一次,和先处理好数据,然后只操作一次dom,这性能上就明显看出差距来了。

下面说一下diff中一个出名的框架snabbdom,vue2中就是使用的它(vue3中已经不用这个了,不过万变不离其宗,学会了snabbdom,对学习diff底层知识很有帮助的)

sanbbdom的github上的地址github.com/snabbdom/sn...

先写一个最简版的

html部分

js 复制代码
<div id="container"></div>

js部分

js 复制代码
import {
  init,
  classModule,
  propsModule,
  styleModule,
  eventListenersModule,
  h
} from "snabbdom";

const patch = init([
  classModule,
  propsModule, 
  styleModule, 
  eventListenersModule
]);

const container = document.getElementById("container");

const vnode = h("h1",{},"我是sanbbdom");
patch(container, vnode)

运行结果

js代码执行完以后,div里的值就变成了我是sanbbdom。

打印一下vnode,看一下控制台都输出了什么

在控制台中,可以看出来,他把节点变成了一个对象,也就是把dom变成了数据

elm是你的节点类型(如div,span的标签名称)

text是文本内容

现在新生成的div只有一层,如果要生成多层级的div,要变一下第三个参数

js 复制代码
const vnode2 = h("div",{},[
  h("p",{},"我是p标签1"),
  h("p",{},"我是p标签2"),
  h("p",{},"我是p标签3")
])
console.log(vnode2)
patch(container, vnode2)

在浏览器中显示

chilren变成了一个数组,里面是三个p标签的数据,text是undefined,因为div里面是三个子元素,不是文字,这就是一个简单的虚拟节点。

在patch方法里,完成了新旧节点的替换,这个替换规则又是怎么替换的,这样分为两种种情况。(这里只针对刚才写的简单patch讨论,复杂的先不涉及)

1 新旧节点的标签名不一样,path方法会先删除旧节点的内容,然后再把新节点的内容插进去。 2 只能通级别比较。

相关推荐
用户05954017446几秒前
把多级缓存一致性验证从手工测试换成 Pytest 参数化,Bug 排查时间缩短 90%
前端·css
暗不需求2 分钟前
深入理解 LangChain:AI 应用开发框架的工程化实践
前端·langchain
用户0595401744623 分钟前
把 Redis 持久化测试从 800 行 Shell 换成 30 行 pytest,排错效率翻了 10 倍
前端·css
GISer_Jing28 分钟前
AI全栈工程师知识体系全景:从前后端核心架构到落地项目全拆解
前端·人工智能·后端·ai编程
Wect33 分钟前
深度剖析浏览器跨域问题
前端·面试·浏览器
陈随易1 小时前
bun将会支持Bun.image,你怎么看?
前端·后端·程序员
jingqingdai31 小时前
别用正则格式化 HTML!我用 DOM 遍历实现零风险本地格式化,老项目重构效率直接拉满
前端·重构·html
木斯佳1 小时前
前端八股文面经大全:腾讯前端实习二、三OC面(2026-04-27)·面经深度解析
前端·状态模式
Python私教1 小时前
如意Agent日志系统重构:从 print() 大海捞针到结构化可观测性栈
java·前端·重构