深入浅出 Vue 中的模板编译

发表于 2年以前  | 总阅读数:1048 次
new Vue({
 render: h => h(App)
})

这个大家都熟悉,调用 render 就会得到传入的模板(.vue文件)对应的虚拟 DOM,那么这个 render 是哪来的呢?它是怎么把 .vue 文件转成浏览器可识别的代码的呢?

render 函数是怎么来的有两种方式

  • 第一种就是经过模板编译生成 render 函数
  • 第二种是我们自己在组件里定义了 render 函数,这种会跳过模板编译的过程

本文将为大家分别介绍这两种,以及详细的编译过程原理

认识模板编译

我们知道 <template></template> 这个是模板,不是真实的 HTML,浏览器是不认识模板的,所以我们需要把它编译成浏览器认识的原生的 HTML

这一块的主要流程就是

  1. 提取出模板中的原生 HTML 和非原生 HTML,比如绑定的属性、事件、指令等等
  2. 经过一些处理生成 render 函数
  3. render 函数再将模板内容生成对应的 vnode
  4. 再经过 patch 过程( Diff )得到要渲染到视图中的 vnode
  5. 最后根据 vnode 创建真实的 DOM 节点,也就是原生 HTML 插入到视图中,完成渲染

上面的 1、2、3 条就是模板编译的过程了

那它是怎么编译,最终生成 render 函数的呢?

模板编译详解——源码

baseCompile()

这就是模板编译的入口函数,它接收两个参数

  • template:就是要转换的模板字符串
  • options:就是转换时需要的参数

编译的流程,主要有三步:

  1. 模板解析:通过正则等方式提取出 <template></template> 模板里的标签元素、属性、变量等信息,并解析成抽象语法树 AST
  2. 优化:遍历 AST 找出其中的静态节点和静态根节点,并添加标记
  3. 代码生成:根据 AST 生成渲染函数 render

这三步分别对应三个函数,后面会一一下介绍,先看一下 baseCompile 源码中是在哪里调用的

源码地址:src/complier/index.js - 11行

export const createCompiler = createCompilerCreator(function baseCompile (
 template: string, // 就是要转换的模板字符串
 options: CompilerOptions //就是转换时需要的参数
): CompiledResult {
 // 1. 进行模板解析,并将结果保存为 AST
 const ast = parse(template.trim(), options)

 // 没有禁用静态优化的话
 if (options.optimize !== false) {
   // 2. 就遍历 AST,并找出静态节点并标记
   optimize(ast, options)
}
 // 3. 生成渲染函数
 const code = generate(ast, options)
 return {
   ast,
   render: code.render, // 返回渲染函数 render
   staticRenderFns: code.staticRenderFns
}
})

就这么几行代码,三步,调用了三个方法很清晰

我们先看一下最后 return 出去的是个啥,再来深入上面这三步分别调用的方法源码,也好更清楚的知道这三步分别是要做哪些处理

编译结果

比如有这样的模板

<template>
   <div id="app">{{name}}</div>
</template>

打印一下编译后的结果,也就是上面源码 return 出去的结果,看看是啥

{
 ast: {
   type: 1,
   tag: 'div',
   attrsList: [ { name: 'id', value: 'app' } ],
   attrsMap: { id: 'app' },
   rawAttrsMap: {},
   parent: undefined,
   children: [
    {
       type: 2,
       expression: '_s(name)',
       tokens: [ { '@binding': 'name' } ],
       text: '{{name}}',
       static: false
    }
  ],
   plain: false,
   attrs: [ { name: 'id', value: '"app"', dynamic: undefined } ],
   static: false,
   staticRoot: false
},
 render: `with(this){return _c('div',{attrs:{"id":"app"}},[_v(_s(name))])}`,
 staticRenderFns: [],
 errors: [],
 tips: []
}

看不明白也没有关系,注意看上面提到的三步都干了啥

  • ast 字段,就是第一步生成的
  • static 字段,就是标记,是在第二步中根据 ast 里的 type 加上去的
  • render 字段,就是第三步生成的

有个大概的印象了,然后再来看源码

1. parse()

源码地址:src/complier/parser/index.js - 79行

就是这个方法就是解析器的主函数,就是它通过正则等方法提取出 <template></template> 模板字符串里所有的 tagpropschildren 信息,生成一个对应结构的 ast 对象

parse 接收两个参数

  • template :就是要转换的模板字符串
  • options:就是转换时需要的参数。它包含有四个钩子函数,就是用来把 parseHTML 解析出来的字符串提取出来,并生成对应的 AST

核心步骤是这样的:

调用 parseHTML 函数对模板字符串进行解析

  • 解析到开始标签、结束标签、文本、注释分别进行不同的处理
  • 解析过程中遇到文本信息就调用文本解析器 parseText 函数进行文本解析
  • 解析过程中遇到包含过滤器,就调用过滤器解析器 parseFilters 函数进行解析

每一步解析的结果都合并到一个对象上(就是最后的 AST)

这个地方的源码实在是太长了,有大几百行代码,我就只贴个大概吧,有兴趣的自己去看一下

export function parse (
 template: string, // 要转换的模板字符串
 options: CompilerOptions // 转换时需要的参数
): ASTElement | void {
 parseHTML(template, {
   warn,
   expectHTML: options.expectHTML,
   isUnaryTag: options.isUnaryTag,
   canBeLeftOpenTag: options.canBeLeftOpenTag,
   shouldDecodeNewlines: options.shouldDecodeNewlines,
   shouldDecodeNewlinesForHref: options.shouldDecodeNewlinesForHref,
   shouldKeepComment: options.comments,
   outputSourceRange: options.outputSourceRange,
   // 解析到开始标签时调用,如 <div>
   start (tag, attrs, unary, start, end) {
       // unary 是否是自闭合标签,如 <img />
       ...
  },
   // 解析到结束标签时调用,如 </div>
   end (tag, start, end) {
       ...
  },
   // 解析到文本时调用
   chars (text: string, start: number, end: number) {
     // 这里会判断判断很多东西,来看它是不是带变量的动态文本
     // 然后创建动态文本或静态文本对应的 AST 节点
     ...
  },
   // 解析到注释时调用
   comment (text: string, start, end) {
     // 注释是这么找的
     const comment = /^<!\--/
     if (comment.test(html)) {
     // 如果是注释,就继续找 '-->'
     const commentEnd = html.indexOf('-->')
     ...
  }
})
 // 返回的这个就是 AST
 return root
}

上面解析文本时调用的 chars() 会根据不同类型节点加上不同 type,来标记 AST 节点类型,这个属性在下一步标记的时候会用到

type AST 节点类型
1 元素节点
2 包含变量的动态文本节点
3 没有变量的纯文本节点

2. optimize()

这个函数就是在 AST 里找出静态节点和静态根节点,并添加标记,为了后面 patch 过程中就会跳过静态节点的对比,直接克隆一份过去,从而优化了 patch 的性能

函数里面调用的外部函数就不贴代码了,大致过程是这样的

  • 标记静态节点(markStatic)。就是判断 type,上面介绍了值为 1、2、3的三种类型

  • type 值为1:就是包含子元素的节点,设置 static 为 false 并递归标记子节点,直到标记完所有子节点

  • type 值为 2:设置 static 为 false

  • type 值为 3:就是不包含子节点和动态属性的纯文本节点,把它的 static = true,patch 的时候就会跳过这个,直接克隆一份去

  • 标记静态根节点(markStaticRoots),这里的原理和标记静态节点基本相同,只是需要满足下面条件的节点才能算作是静态根节点

  • 节点本身必须是静态节点

  • 必须有子节点

  • 子节点不能只有一个文本节点

源码地址:src/complier/optimizer.js - 21行

export function optimize (root: ?ASTElement, options: CompilerOptions) {
 if (!root) return
 isStaticKey = genStaticKeysCached(options.staticKeys || '')
 isPlatformReservedTag = options.isReservedTag || no
 // 标记静态节点
 markStatic(root)
 // 标记静态根节点
 markStaticRoots(root, false)
}

3. generate()

这个就是生成 render 的函数,就是说最终会返回下面这样的东西

// 比如有这么个模板
<template>
   <div id="app">{{name}}</div>
</template>

// 上面模板编译后返回的 render 字段 就是这样的
render: `with(this){return _c('div',{attrs:{"id":"app"}},[_v(_s(name))])}`

// 把内容格式化一下,容易理解一点
with(this){
 return _c(
   'div',
  { attrs:{"id":"app"} },
  [  _v(_s(name)) ]
)
}

这个结构是不是有点熟悉?

了解虚拟 DOM 就可以看出来,上面的 render 正是虚拟 DOM 的结构,就是把一个标签分为 tagpropschildren,没有错

在看 generate 源码之前,我们要先了解一下上面这最后返回的 render 字段是什么意思,再来看 generate 源码,就会轻松得多,不然连函数返回的东西是干嘛的都不知道怎么可能看得懂这个函数呢

render

我们来翻译一下上面编译出来的 render

这个 with 在 《你不知道的JavaScript》上卷里介绍的是,用来欺骗词法作用域的关键字,它可以让我们更快的引用一个对象上的多个属性

看个例子

const name = '掘金'
const obj = { name:'沐华', age: 18 }
with(obj){
   console.log(name) // 沐华 不需要写 obj.name 了
   console.log(age) // 18   不需要写 obj.age 了
}

上面的 with(this){} 里的 this 就是当前组件实例。因为通过 with 改变了词法作用域中属性的指向,所以标签里使用 name 直接用就是了,而不需要 this.name 这样

_c_v_s 是什么呢?

在源码里是这样定义的,格式是:_c(缩写) = createElement(函数名)

源码地址:src/core/instance/render-helpers/index.js - 15行

// 其实不止这几个,由于本文例子中没有用到就没都复制过来占位了
export function installRenderHelpers (target: any) {
 target._s = toString // 转字符串函数
 target._l = renderList // 生成列表函数
 target._v = createTextVNode // 创建文本节点函数
 target._e = createEmptyVNode // 创建空节点函数
}
// 补充
_c = createElement // 创建虚拟节点函数

再来看是不是就清楚多了呢

with(this){ // 欺骗词法作用域,将该作用域里所有属姓和方法都指向当前组件
 return _c( // 创建一个虚拟节点
   'div', // 标签为 div
  { attrs:{"id":"app"} }, // 有一个属性 id 为 'app'
  [  _v(_s(name)) ] // 是一个文本节点,所以把获取到的动态属性 name 转成字符串
)
}

接下来我们再来看 generate() 源码

generate

源码地址:src/complier/codegen/index.js - 43行

这个流程很简单,只有几行代码,就是先判断 AST 是不是为空,不为空就根据 AST 创建 vnode,否则就创建一个空div 的 vnode

export function generate (
 ast: ASTElement | void,
 options: CompilerOptions
): CodegenResult {
 const state = new CodegenState(options)
 // 就是先判断 AST 是不是为空,不为空就根据 AST 创建 vnode,否则就创建一个空div的 vnode
 const code = ast ? (ast.tag === 'script' ? 'null' : genElement(ast, state)) : '_c("div")'

 return {
   render: `with(this){return ${code}}`,
   staticRenderFns: state.staticRenderFns
}
}

可以看出这里面主要就是通过 genElement() 方法来创建 vnode 的,所以我们来看一下它的源码,看是怎么创建的

genElement()

源码地址:src/complier/codegen/index.js - 56行

这里的逻辑还是很清晰的,就是一堆 if/else 判断传进来的 AST 元素节点的属性来执行不同的生成函数

这里还可以发现另一个知识点 v-for 的优先级要高于 v-if,因为先判断 for 的

export function genElement (el: ASTElement, state: CodegenState): string {
 if (el.parent) {
   el.pre = el.pre || el.parent.pre
}

 if (el.staticRoot && !el.staticProcessed) {
   return genStatic(el, state)
} else if (el.once && !el.onceProcessed) { // v-once
   return genOnce(el, state)
} else if (el.for && !el.forProcessed) { // v-for
   return genFor(el, state)
} else if (el.if && !el.ifProcessed) { // v-if
   return genIf(el, state)

   // template 节点 && 没有插槽 && 没有 pre 标签
} else if (el.tag === 'template' && !el.slotTarget && !state.pre) {
   return genChildren(el, state) || 'void 0'
} else if (el.tag === 'slot') { // v-slot
   return genSlot(el, state)
} else {
   // component or element
   let code
   // 如果有子组件
   if (el.component) {
     code = genComponent(el.component, el, state)
  } else {
     let data
     // 获取元素属性 props
     if (!el.plain || (el.pre && state.maybeComponent(el))) {
       data = genData(el, state)
    }
     // 获取元素子节点
     const children = el.inlineTemplate ? null : genChildren(el, state, true)
     code = `_c('${el.tag}'${
       data ? `,${data}` : '' // data
     }${
       children ? `,${children}` : '' // children
     })`
  }
   // module transforms
   for (let i = 0; i < state.transforms.length; i++) {
     code = state.transforms[i](el, code)
  }
   // 返回上面作为 with 作用域执行的内容
   return code
}
}

每一种类型调用的生成函数就不一一列举了,总的来说最后创建出来的 vnode 节点类型无非就三种,元素节点、文本节点、注释节点

自定义的 render

先举个例子吧,三种情况如下

// 1. test.vue
<template>
   <h1>我是沐华</h1>
</template>
<script>
 export default {}
</script>
// 2. test.vue
<script>
 export default {
   render(h){
     return h('h1',{},'我是沐华')
  }
}
</script>
// 3. test.js
export default {
 render(h){
   return h('h1',{},'我是沐华')
}
}

上面三种,最后渲染的出来的就是完全一模一样的,因为这个 h 就是上面模板编译后的那个 _c

这时有人可能就会问,为什么要自己写呢,不是有模板编译自动生成吗?

这个问题问得好!自己写肯定是有好处的

  1. 自己把 vnode 给写了,就会直接跳过了模板编译,不用去解析模板里的动态属性、事件、指令等等了,所以性能上会有那么一丢丢提升。这一点在下面的渲染的优先级上就有体现
  2. 还有一些情况,能让我们代码写法的更加灵活,更加方便简洁,不会冗余

比如 Element-UI 里面的组件源码里就有大量直接写 render 函数

接下来分别看下这两点是如何体现的

1. 渲染优先级

先看一下在官网的生命周期里,关于模板编译的部分

如图可以知道,如果有 template,就不会管 el 了,所以 template 比 el 的优先级更高,比如

那我们自己写了 render 呢?

<div id='app'>
   <p>{{ name }}</p>
</div>
<script>
   new Vue({
       el:'#app',
       data:{ name:'沐华' },
       template:'<div>掘金</div>',
       render(h){
           return h('div', {}, '好好学习,天天向上')
      }
  })
</script>

这个代码执行后页面渲染出来只有 <div>好好学习,天天向上</div>

可以得出 render 函数的优先级更高

因为不管是 el 挂载的,还是 emplate 最后都会被编译成 render 函数,而如果已经有了 render 函数了,就跳过前面的编译了

这一点在源码里也有体现

在源码中找到答案:dist/vue.js - 11927行

 Vue.prototype.$mount = function ( el, hydrating ) {
   el = el && query(el);
   var options = this.$options;
   // 如果没有 render
   if (!options.render) {
     var template = options.template;
     // 再判断,如果有 template
     if (template) {
       if (typeof template === 'string') {
         if (template.charAt(0) === '#') {
           template = idToTemplate(template);
        }
      } else if (template.nodeType) {
         template = template.innerHTML;
      } else {
         return this
      }
     // 再判断,如果有 el
    } else if (el) {
       template = getOuterHTML(el);
    }
  }
   return mount.call(this, el, hydrating)
};

2. 更灵活的写法

比如说我们需要写很多 if 判断的时候

<template>
   <h1 v-if="level === 1">
     <a href="xxx">
       <slot></slot>
     </a>
   </h1>
   <h2 v-else-if="level === 2">
     <a href="xxx">
       <slot></slot>
     </a>
   </h2>
   <h3 v-else-if="level === 3">
     <a href="xxx">
       <slot></slot>
     </a>
   </h3>
</template>
<script>
 export default {
   props:['level']
}
</script>

不知道你有没有写过类似上面这样的代码呢?

我们换一种方式来写出和上面一模一样的代码看看,直接写 render

<script>
 export default {
   props:['level'],
   render(h){
     return h('h' + this.level, this.$slots.default())
  }
}
</script>

搞定!就这!就这?

没错,就这!

或者下面这样,多次调用的时候就很方便

<script>
 export default {
   props:['level'],
   render(h){
     const tag = 'h' + this.level
     return (<tag>{this.$slots.default()}</tag>)
  }
}
</script>

补充

如果想知道更多格式的模板编译出来是什么样的,可以这样

Vue2 的模板编译可以安装 vue-template-compiler

Vue3 的模板编译可以点这里

然后自行测试

另外在 Vue3 里模板编译部分有一些修改,可以点击下面链接,看下 深入浅出虚拟 DOM 和 Diff 算法,里面有介绍

结语

如果本文对你有一丁点帮助,点个赞支持一下吧,感谢感谢

本文由哈喽比特于2年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/odrnJhKUa5W0n8w0Ul4gaA

 相关推荐

刘强东夫妇:“移民美国”传言被驳斥

京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。

发布于:7月以前  |  808次阅读  |  详细内容 »

博主曝三大运营商,将集体采购百万台华为Mate60系列

日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。

发布于:7月以前  |  770次阅读  |  详细内容 »

ASML CEO警告:出口管制不是可行做法,不要“逼迫中国大陆创新”

据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。

发布于:7月以前  |  756次阅读  |  详细内容 »

抖音中长视频App青桃更名抖音精选,字节再发力对抗B站

今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。

发布于:7月以前  |  648次阅读  |  详细内容 »

威马CDO:中国每百户家庭仅17户有车

日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。

发布于:7月以前  |  589次阅读  |  详细内容 »

研究发现维生素 C 等抗氧化剂会刺激癌症生长和转移

近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。

发布于:7月以前  |  449次阅读  |  详细内容 »

苹果据称正引入3D打印技术,用以生产智能手表的钢质底盘

据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。

发布于:7月以前  |  446次阅读  |  详细内容 »

千万级抖音网红秀才账号被封禁

9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...

发布于:7月以前  |  445次阅读  |  详细内容 »

亚马逊股东起诉公司和贝索斯,称其在购买卫星发射服务时忽视了 SpaceX

9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。

发布于:7月以前  |  444次阅读  |  详细内容 »

苹果上线AppsbyApple网站,以推广自家应用程序

据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。

发布于:7月以前  |  442次阅读  |  详细内容 »

特斯拉美国降价引发投资者不满:“这是短期麻醉剂”

特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。

发布于:7月以前  |  441次阅读  |  详细内容 »

光刻机巨头阿斯麦:拿到许可,继续对华出口

据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。

发布于:7月以前  |  437次阅读  |  详细内容 »

马斯克与库克首次隔空合作:为苹果提供卫星服务

近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。

发布于:7月以前  |  430次阅读  |  详细内容 »

𝕏(推特)调整隐私政策,可拿用户发布的信息训练 AI 模型

据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。

发布于:7月以前  |  428次阅读  |  详细内容 »

荣耀CEO谈华为手机回归:替老同事们高兴,对行业也是好事

9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。

发布于:7月以前  |  423次阅读  |  详细内容 »

AI操控无人机能力超越人类冠军

《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。

发布于:7月以前  |  423次阅读  |  详细内容 »

AI生成的蘑菇科普书存在可致命错误

近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。

发布于:7月以前  |  420次阅读  |  详细内容 »

社交媒体平台𝕏计划收集用户生物识别数据与工作教育经历

社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”

发布于:7月以前  |  411次阅读  |  详细内容 »

国产扫地机器人热销欧洲,国产割草机器人抢占欧洲草坪

2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。

发布于:7月以前  |  406次阅读  |  详细内容 »

罗永浩吐槽iPhone15和14不会有区别,除了序列号变了

罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。

发布于:7月以前  |  398次阅读  |  详细内容 »
 目录