Vue.js中template编译问题如何解决

本文主要介绍vue.jstemplate编译的问题,给大家做个参考,希望大家学完本文对vue.js中template编译问题就有清晰的解决思路了。

写在前面

因为对Vue.js很感兴趣,而且平时工作的技术栈也是Vue.js,这几个月花了些时间研究学习了一下Vue.js源码,并做了总结与输出。

文章的原地址:https://github.com/answershuto/learnVue。

在学习过程中,为Vue加上了中文的注释https://github.com/answershuto/learnVue/tree/master/vue-src,希望可以对其他想学习Vue源码的小伙伴有所帮助。

立即学习“前端免费学习笔记(深入)”;

可能会有理解存在偏差的地方,欢迎提issue指出,共同学习,共同进步。

$mount

首先看一下mount的代码

/*把原本不带编译的$mount方法保存下来,在最后会调用。*/const mount = Vue.prototype.$mount/*挂载组件,带模板编译*/Vue.prototype.$mount = function ( el?: string | Element, hydrating?: boolean): Component { el = el && query(el) /* istanbul ignore if */ if (el === document.body || el === document.documentElement) {  process.env.NODE_ENV !== 'production' && warn(   `Do not mount Vue to  or  - mount to normal elements instead.`  )  return this } const options = this.$options // resolve template/el and convert to render function /*处理模板templete,编译成render函数,render不存在的时候才会编译template,否则优先使用render*/ if (!options.render) {  let template = options.template  /*template存在的时候取template,不存在的时候取el的outerHTML*/  if (template) {   /*当template是字符串的时候*/   if (typeof template === 'string') {    if (template.charAt(0) === '#') {     template = idToTemplate(template)     /* istanbul ignore if */     if (process.env.NODE_ENV !== 'production' && !template) {      warn(       `Template element not found or is empty: ${options.template}`,       this      )     }    }   } else if (template.nodeType) {    /*当template为DOM节点的时候*/    template = template.innerHTML   } else {    /*报错*/    if (process.env.NODE_ENV !== 'production') {     warn('invalid template option:' + template, this)    }    return this   }  } else if (el) {   /*获取element的outerHTML*/   template = getOuterHTML(el)  }  if (template) {   /* istanbul ignore if */   if (process.env.NODE_ENV !== 'production' && config.performance && mark) {    mark('compile')   }   /*将template编译成render函数,这里会有render以及staticRenderFns两个返回,这是vue的编译时优化,static静态不需要在VNode更新时进行patch,优化性能*/   const { render, staticRenderFns } = compileToFunctions(template, {    shouldDecodeNewlines,    delimiters: options.delimiters   }, this)   options.render = render   options.staticRenderFns = staticRenderFns   /* istanbul ignore if */   if (process.env.NODE_ENV !== 'production' && config.performance && mark) {    mark('compile end')    measure(`${this._name} compile`, 'compile', 'compile end')   }  } } /*Github:https://github.com/answershuto*/ /*调用const mount = Vue.prototype.$mount保存下来的不带编译的mount*/ return mount.call(this, el, hydrating)}

登录后复制

通过mount代码我们可以看到,在mount的过程中,如果render函数不存在(render函数存在会优先使用render)会将template进行compileToFunctions得到render以及staticRenderFns。譬如说手写组件时加入了template的情况都会在运行时进行编译。而render function在运行后会返回VNode节点,供页面的渲染以及在update的时候patch。接下来我们来看一下template是如何编译的。

一些基础

首先,template会被编译成AST语法树,那么AST是什么?

在计算机科学中,抽象语法树(abstract syntax tree或者缩写为AST),或者语法树(syntax tree),是源代码的抽象语法结构的树状表现形式,这里特指编程语言的源代码。

AST会经过generate得到render函数,render的返回值是VNode,VNode是Vue的虚拟DOM节点,具体定义如下:

export default class VNode { tag: string | void; data: VNodeData | void; children: ?Array; text: string | void; elm: Node | void; ns: string | void; context: Component | void; // rendered in this component's scope functionalContext: Component | void; // only for functional component root nodes key: string | number | void; componentOptions: VNodeComponentOptions | void; componentInstance: Component | void; // component instance parent: VNode | void; // component placeholder node 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? /*Github:https://github.com/answershuto*/  constructor (  tag?: string,  data?: VNodeData,  children?: ?Array,  text?: string,  elm?: Node,  context?: Component,  componentOptions?: VNodeComponentOptions ) {  /*当前节点的标签名*/  this.tag = tag  /*当前节点对应的对象,包含了具体的一些数据信息,是一个VNodeData类型,可以参考VNodeData类型中的数据信息*/  this.data = data  /*当前节点的子节点,是一个数组*/  this.children = children  /*当前节点的文本*/  this.text = text  /*当前虚拟节点对应的真实dom节点*/  this.elm = elm  /*当前节点的名字空间*/  this.ns = undefined  /*编译作用域*/  this.context = context  /*函数化组件作用域*/  this.functionalContext = undefined  /*节点的key属性,被当作节点的标志,用以优化*/  this.key = data && data.key  /*组件的option选项*/  this.componentOptions = componentOptions  /*当前节点对应的组件的实例*/  this.componentInstance = undefined  /*当前节点的父节点*/  this.parent = undefined  /*简而言之就是是否为原生HTML或只是普通文本,innerHTML的时候为true,textContent的时候为false*/  this.raw = false  /*静态节点标志*/  this.isStatic = false  /*是否作为跟节点插入*/  this.isRootInsert = true  /*是否为注释节点*/  this.isComment = false  /*是否为克隆节点*/  this.isCloned = false  /*是否有v-once指令*/  this.isOnce = false } // DEPRECATED: alias for componentInstance for backwards compat. /* istanbul ignore next */ get child (): Component | void {  return this.componentInstance }}

登录后复制

关于VNode的一些细节,请参考VNode节点。

createCompiler

createCompiler用以创建编译器,返回值是compile以及compileToFunctions。compile是一个编译器,它会将传入的template转换成对应的AST树、render函数以及staticRenderFns函数。而compileToFunctions则是带缓存的编译器,同时staticRenderFns以及render函数会被转换成Funtion对象。

因为不同平台有一些不同的options,所以createCompiler会根据平台区分传入一个baseOptions,会与compile本身传入的options合并得到最终的finalOptions。

compileToFunctions

首先还是贴一下compileToFunctions的代码。

 /*带缓存的编译器,同时staticRenderFns以及render函数会被转换成Funtion对象*/ function compileToFunctions (  template: string,  options?: CompilerOptions,  vm?: Component ): CompiledFunctionResult {  options = options || {}  /* istanbul ignore if */  if (process.env.NODE_ENV !== 'production') {   // detect possible CSP restriction   try {    new Function('return 1')   } catch (e) {    if (e.toString().match(/unsafe-eval|CSP/)) {     warn(      'It seems you are using the standalone build of Vue.js in an ' +      'environment with Content Security Policy that prohibits unsafe-eval. ' +      'The template compiler cannot work in this environment. Consider ' +      'relaxing the policy to allow unsafe-eval or pre-compiling your ' +      'templates into render functions.'     )    }   }  }  /*Github:https://github.com/answershuto*/  // check cache  /*有缓存的时候直接取出缓存中的结果即可*/  const key = options.delimiters   ? String(options.delimiters) + template   : template  if (functionCompileCache[key]) {   return functionCompileCache[key]  }  // compile  /*编译*/  const compiled = compile(template, options)  // check compilation errors/tips  if (process.env.NODE_ENV !== 'production') {   if (compiled.errors && compiled.errors.length) {    warn(     `Error compiling template:${template}` +     compiled.errors.map(e => `- ${e}`).join('') + '',     vm    )   }   if (compiled.tips && compiled.tips.length) {    compiled.tips.forEach(msg => tip(msg, vm))   }  }  // turn code into functions  const res = {}  const fnGenErrors = []  /*将render转换成Funtion对象*/  res.render = makeFunction(compiled.render, fnGenErrors)  /*将staticRenderFns全部转化成Funtion对象 */  const l = compiled.staticRenderFns.length  res.staticRenderFns = new Array(l)  for (let i = 0; i  `${err.toString()} in$[code]`).join(''),     vm    )   }  }  /*存放在缓存中,以免每次都重新编译*/  return (functionCompileCache[key] = res)  }

登录后复制

我们可以发现,在闭包中,会有一个functionCompileCache对象作为缓存器。

 /*作为缓存,防止每次都重新编译*/ const functionCompileCache: {  [key: string]: CompiledFunctionResult; } = Object.create(null)

登录后复制

在进入compileToFunctions以后,会先检查缓存中是否有已经编译好的结果,如果有结果则直接从缓存中读取。这样做防止每次同样的模板都要进行重复的编译工作。

  // check cache  /*有缓存的时候直接取出缓存中的结果即可*/  const key = options.delimiters   ? String(options.delimiters) + template   : template  if (functionCompileCache[key]) {   return functionCompileCache[key]  }

登录后复制

在compileToFunctions的末尾会将编译结果进行缓存

 /*存放在缓存中,以免每次都重新编译*/ return (functionCompileCache[key] = res)

登录后复制

 compile

 /*编译,将模板template编译成AST树、render函数以及staticRenderFns函数*/ function compile (  template: string,  options?: CompilerOptions ): CompiledResult {  const finalOptions = Object.create(baseOptions)  const errors = []  const tips = []  finalOptions.warn = (msg, tip) => {   (tip ? tips : errors).push(msg)  }  /*做下面这些merge的目的因为不同平台可以提供自己本身平台的一个baseOptions,内部封装了平台自己的实现,然后把共同的部分抽离开来放在这层compiler中,所以在这里需要merge一下*/  if (options) {   // merge custom modules   /*合并modules*/   if (options.modules) {    finalOptions.modules = (baseOptions.modules || []).concat(options.modules)   }   // merge custom directives   if (options.directives) {    /*合并directives*/    finalOptions.directives = extend(     Object.create(baseOptions.directives),     options.directives    )   }   // copy other options   for (const key in options) {    /*合并其余的options,modules与directives已经在上面做了特殊处理了*/    if (key !== 'modules' && key !== 'directives') {     finalOptions[key] = options[key]    }   }  }  /*基础模板编译,得到编译结果*/  const compiled = baseCompile(template, finalOptions)  if (process.env.NODE_ENV !== 'production') {   errors.push.apply(errors, detectErrors(compiled.ast))  }  compiled.errors = errors  compiled.tips = tips  return compiled }

登录后复制

compile主要做了两件事,一件是合并option(前面说的将平台自有的option与传入的option进行合并),另一件是baseCompile,进行模板template的编译。

来看一下baseCompile

baseCompile

function baseCompile ( template: string, options: CompilerOptions): CompiledResult { /*parse解析得到ast树*/ const ast = parse(template.trim(), options) /*  将AST树进行优化  优化的目标:生成模板AST树,检测不需要进行DOM改变的静态子树。  一旦检测到这些静态树,我们就能做以下这些事情:  1.把它们变成常数,这样我们就再也不需要每次重新渲染时创建新的节点了。  2.在patch的过程中直接跳过。 */ optimize(ast, options) /*根据ast树生成所需的code(内部包含render与staticRenderFns)*/ const code = generate(ast, options) return {  ast,  render: code.render,  staticRenderFns: code.staticRenderFns }}

登录后复制

baseCompile首先会将模板template进行parse得到一个AST语法树,再通过optimize做一些优化,最后通过generate得到render以及staticRenderFns。

parse

parse的源码可以参见https://github.com/answershuto/learnVue/blob/master/vue-src/compiler/parser/index.js#L53。

parse会用正则等方式解析template模板中的指令、class、style等数据,形成AST语法树。

optimize

optimize的主要作用是标记static静态节点,这是Vue在编译过程中的一处优化,后面当update更新界面时,会有一个patch的过程,diff算法会直接跳过静态节点,从而减少了比较的过程,优化了patch的性能。

generate

generate是将AST语法树转化成render funtion字符串的过程,得到结果是render的字符串以及staticRenderFns字符串。

至此,我们的template模板已经被转化成了我们所需的AST语法树、render function字符串以及staticRenderFns字符串。

举个例子

来看一下这段代码的编译结果

{{text}}

hello world

{{item.name}}

{{item.value}}

{{index}}

---

{{text}}

登录后复制

转化后得到AST树,如下图:

Vue.js中template编译问题如何解决

我们可以看到最外层的p是这颗AST树的根节点,节点上有许多数据代表这个节点的形态,比如static表示是否是静态节点,staticClass表示静态class属性(非bind:class)。children代表该节点的子节点,可以看到children是一个长度为4的数组,里面包含的是该节点下的四个p子节点。children里面的节点与父节点的结构类似,层层往下形成一棵AST语法树。

再来看看由AST得到的render函数

with(this){  return _c( 'p',        {          /*static class*/          staticClass:"main",          /*bind class*/          class:bindClass        },        [          _c( 'p', [_v(_s(text))]),          _c('p',[_v("hello world")]),          /*这是一个v-for循环*/          _l(            (arr),            function(item,index){              return _c( 'p',                    [_c('p',[_v(_s(item.name))]),                    _c('p',[_v(_s(item.value))]),                    _c('p',[_v(_s(index))]),                    _c('p',[_v("---")])]                  )            }          ),          /*这是v-if*/          (text)?_c('p',[_v(_s(text))]):_c('p',[_v("no text")])],          2      )}

登录后复制

_c,_v,_s,_q

看了render function字符串,发现有大量的_c,_v,_s,_q,这些函数究竟是什么?

带着问题,我们来看一下core/instance/render。

/*处理v-once的渲染函数*/ Vue.prototype._o = markOnce /*将字符串转化为数字,如果转换失败会返回原字符串*/ Vue.prototype._n = toNumber /*将val转化成字符串*/ Vue.prototype._s = toString /*处理v-for列表渲染*/ Vue.prototype._l = renderList /*处理slot的渲染*/ Vue.prototype._t = renderSlot /*检测两个变量是否相等*/ Vue.prototype._q = looseEqual /*检测arr数组中是否包含与val变量相等的项*/ Vue.prototype._i = looseIndexOf /*处理static树的渲染*/ Vue.prototype._m = renderStatic /*处理filters*/ Vue.prototype._f = resolveFilter /*从config配置中检查eventKeyCode是否存在*/ Vue.prototype._k = checkKeyCodes /*合并v-bind指令到VNode中*/ Vue.prototype._b = bindObjectProps /*创建一个文本节点*/ Vue.prototype._v = createTextVNode /*创建一个空VNode节点*/ Vue.prototype._e = createEmptyVNode /*处理ScopedSlots*/ Vue.prototype._u = resolveScopedSlots /*创建VNode节点*/ vm._c = (a, b, c, d) => createElement(vm, a, b, c, d, false)

登录后复制

通过这些函数,render函数最后会返回一个VNode节点,在_update的时候,经过patch与之前的VNode节点进行比较,得出差异后将这些差异渲染到真实的DOM上。

相关推荐:

微信小程序template模板详解

微信小程序的template模板如何使用

HTML5中标签的详细介绍(图文)

以上就是Vue.js中template编译问题如何解决的详细内容,更多请关注【创想鸟】其它相关文章!

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

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2788792.html

(0)
上一篇 2025年3月8日 18:31:24
下一篇 2025年2月27日 02:55:01

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • Typescript开发node.js项目实例详解

    本文主要介绍了详解使用typescript开发node.js项目(简单的环境配置),非常具有实用价值,需要的朋友可以参考下,希望能帮助到大家。 最近在学习typescript的过程中,想到也许可以使用ts来开发node.js项目。在网上搜了…

    2025年3月8日
    200
  • 关于JS中Attr的用法

    本文主要通过实例代码给大家介绍了js中的attr的用法,非常不错,具有参考借鉴价值,需要的朋友参考下吧,希望能帮助到大家。 具体代码如下所示: $(document).ready(function(){ $(“#btn”).click(fu…

    编程技术 2025年3月8日
    200
  • JS实现汉字按拼音排序的方法

    本文主要介绍了js实现中文汉字按拼音排序的方法,涉及javascript针对中文字符串的转换、遍历、排序等相关操作技巧,需要的朋友可以参考下,希望能帮助到大家。 代码1,拼音排序: var array = [‘武汉’, ‘北京’, ‘上海’…

    编程技术 2025年3月8日
    200
  • JavaScript正则表达式常用基本语法

    本文来给大家讲解一下正则表达式常用基本语法,学习正则表达式的常用基本语法是学习编程的基础,正则表达式在程序中应用非常广泛,在我们在做验证的时候正则用的最为明显,本篇文章会带着大家来学习了解一下正则表达式的常用基本语法 1.正则表达式基本语法…

    编程技术 2025年3月8日
    200
  • js动态逐行添加、删除、遍历取值

    关于js对表格进行逐行添加,今天抽空整理了一下:新建一个html文件(没有编辑器的可以新建一个demo.txt文件,然后改后缀名为demo.html),把下面代码全部贴进去即可。 功能包括:表格添加一行,表格删除一行,表格遍历取值等。 点击…

    2025年3月8日
    200
  • js给datalist或select动态添加option详解

    有时需要从配置文件中读取信息,然后添加到用户的选择项中,比如select的option选项,本文主要为大家分享一篇利用js给datalist或select动态添加option选项的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过…

    编程技术 2025年3月8日
    200
  • nodejs连接MySQL数据库如何实现

    本文主要介绍了nodejs实现的连接mysql数据库功能,结合实例形式分析了nodejs连接及查询mysql数据的相关操作步骤与实现技巧,需要的朋友可以参考下,希望能帮助到大家。 1、在工程目录下运行npm install mysql安装用…

    编程技术 2025年3月8日
    200
  • JavaScript职责链模式实例方法

    本文主要介绍了javascript实现职责链模式概述,详细的介绍了什么是职责链模式和实现方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能帮助到大家。 什么是职责链模式 职责链模式的定义是:使多个对象都有机会处理请求,从而避免请…

    编程技术 2025年3月8日
    200
  • JS全排列组合算法实现方法

    全排列组合算法,例如a,b,c,d进行全排列组合,则组合结果为:a,b,ab,c,ac,bc,abc,d,ad,bd,abd,cd,acd,bcd,abcd。实现思路:从数据源拿出一个元素,依次与已存在的组合数据进行组合,循环上面操作直到数…

    2025年3月8日
    200
  • js和css标签内容切换功能的实现教程

    本文主要和大家分享js + css实现标签内容切换功能(实例讲解)。我们先附上效果图,并用实例代码教大家,希望能帮助到大家。 先附上效果图和代码: html 文档: nbsp;html>  Title   window.onload …

    2025年3月8日
    200

发表回复

登录后才能评论