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 只能通级别比较。

相关推荐
jerrywus4 分钟前
前端老哥的救命稻草:用 Obsidian 搞定 Claude Code 的「金鱼记忆」
前端·agent·claude
球球pick小樱花9 分钟前
游戏官网前端工具库:海内外案例解析
前端·javascript·css
用户605723748730817 分钟前
AI 编码助手的规范驱动开发 - OpenSpec 初探
前端·后端·程序员
狗胜19 分钟前
AI观察日记 2026-03-02|CLAUDE、TYPE、APPFUNCTIONS:掘金热门里的下一步信号
前端
喝水的长颈鹿21 分钟前
【大白话前端 02】网页从解析到绘制的全流程
前端·javascript
明君8799723 分钟前
Flutter 实现 AI 聊天页面 —— 记一次 Markdown 数学公式显示的踩坑之旅
前端·flutter
用户145369814587824 分钟前
VersionCheck.js - 让前端版本更新变得简单优雅
前端·javascript
米饭同学i25 分钟前
微信小程序实现随机撒花效果
前端
Arthur147261228654726 分钟前
模块化和组件化的区别
前端
codingWhat28 分钟前
整理「祖传」代码,就是在开发脚手架?
前端·javascript·node.js