Vue源码之虚拟DOM长成啥样

Posted 天地会珠海分舵

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Vue源码之虚拟DOM长成啥样相关的知识,希望对你有一定的参考价值。

这些年写前端vue时经常碰到虚拟DOM这个单词,但都是听到就算了,左耳进右耳出,知道有这么个东西就完了,因为感觉也没有影响我实现公司的产品业务逻辑。

今天在这里想花点时间探讨下虚拟DOM相关的知识,给自己补补课。

1. 一个简单的虚拟DOM长什么样

其实当今前端框架不少用到了虚拟DOM的技术,但各家有各家的实现。这里我们先看下比较简单的虚拟DOM库snabbdom的虚拟DOM长什么样

我们假设有html如下,其实也就是所谓的真实DOM

<div class="message">Hello World</div>

那么snabbdom对应的虚拟DOM是长以下这个样子的:


    "sel": "div",
    "data": 
        "class": "message"
    ,
    "text": "Hello World"

这样看来一个虚拟DOM其实就是一个json文件,里面的内容也不难理解,猜都能猜出来:

  • sel: 标签名
  • data: 标签属性
  • text: 标签之间的文本内容

从这里就不难理解虚拟DOM和真实DOM的关系,虚拟DOM就是一个用来描述真实DOM节点的javascript对象。

虚拟DOM除了可以描述单一的一个真实DOM节点,还能描述一颗DOM数。

比如有html如下:

`<div id="container">
  <div class="message">Hello World</div>
</div>

对应的虚拟DOM将会是


    "sel": "div",
    "data": 
        "id": "container"
    ,
    "children": [
        
            "sel": "div",
            "data": 
                "class": "message"
            ,
            "text": "Hello World"
        
    ]

虚拟DOM对象里面多了个children的数组选项,里面的内容就是前面html中的子节点。

2. Vue中的虚拟DOM长什么样

还是以最简单的html节点为例子

<div class="message">Hello World</div>

vue中的虚拟DOM将会是如下这个样子的



  tag: "div",
  data: 
      "staticClass": "message"
  
  children: [
    
      text: "Hello World",
      ...
    ,
  ],
  ...

Vue中的虚拟DOM比snabbdom的虚拟DOM复杂,会多出很多属性,我这里就不一一列出来了,上面只是显示了一些我觉得和snabbdom的虚拟DOM差不多的属性。

两个虚拟DOM看上去差不多,只是:

  • 标签名这里用tag而非sel
  • 标签之间的文本在vue中用一个文本虚拟子节点来表示,而不是像snabbdom那样直接放到自身的text中

3. Vue中的虚拟DOM实现

在Vue中,虚拟DOM是通过VNode这个类来实现的。

export default class VNode 
  tag: string | void;
  data: VNodeData | void;
  children: ?Array<VNode>;
  text: string | void;
  elm: Node | void;
  ns: string | void;
  context: Component | void; // rendered in this component's scope
  key: string | number | void;
  componentOptions: VNodeComponentOptions | void;
  componentInstance: Component | void; // component instance
  parent: VNode | void; // component placeholder node

  // strictly internal
  raw: boolean; // contains raw HTML? (server only)
  isStatic: boolean; // hoisted static node
  isRootInsert: boolean; // necessary for enter transition check
  isComment: boolean; // empty comment placeholder?
  isCloned: boolean; // is a cloned node?
  isOnce: boolean; // is a v-once node?
  asyncFactory: Function | void; // async component factory function
  asyncMeta: Object | void;
  isAsyncPlaceholder: boolean;
  ssrContext: Object | void;
  fnContext: Component | void; // real context vm for functional nodes
  fnOptions: ?ComponentOptions; // for SSR caching
  devtoolsMeta: ?Object; // used to store functional render context for devtools
  fnScopeId: ?string; // functional scope id support

  constructor(
    tag?: string,
    data?: VNodeData,
    children?: ?Array<VNode>,
    text?: string,
    elm?: Node,
    context?: Component,
    componentOptions?: VNodeComponentOptions,
    asyncFactory?: Function
  ) 
    this.tag = tag;
    this.data = data;
    this.children = children;
    this.text = text;
    this.elm = elm;
    this.ns = undefined;
    this.context = context;
    this.fnContext = undefined;
    this.fnOptions = undefined;
    this.fnScopeId = undefined;
    this.key = data && data.key;
    this.componentOptions = componentOptions;
    this.componentInstance = undefined;
    this.parent = undefined;
    this.raw = false;
    this.isStatic = false;
    this.isRootInsert = true;
    this.isComment = false;
    this.isCloned = false;
    this.isOnce = false;
    this.asyncFactory = asyncFactory;
    this.asyncMeta = undefined;
    this.isAsyncPlaceholder = false;
  

其中包含了我们上面示例看到的最重要的tag,data,children等描述一个DOM结构的选项,还有很多Vue实现需要用到的选项,这里就不一一解析,什么时候用到什么时候分析吧,先有个概念就好了。

4. createTextVNode

从VNode的构造函数可以看到其接受的参数比较多,为了方便使用,vue为创建VNode节点提供了一些函数的封装,其中我们最常用的就是创建一个文本节点。

export function createTextVNode(val: string | number) 
  return new VNode(undefined, undefined, undefined, String(val));

对应的真实DOM

<div>Hello world</div>

我是@天地会珠海分舵,「青葱日历」和「三日清单」 作者。能力一般,水平有限,觉得我说的还有那么点道理的不妨点个赞关注下!

以上是关于Vue源码之虚拟DOM长成啥样的主要内容,如果未能解决你的问题,请参考以下文章

Vue源码之虚拟DOM来自何方?

Vue源码之虚拟DOM来自何方?

Vue源码之虚拟DOM

笔记Vue源码解析之虚拟DOM和diff算法

面试中的网红Vue源码解析之虚拟DOM,你知多少呢?深入解读diff算法

Vue源码虚拟DOM将去往何处?