vue parseHTML函数源码解析 AST预备知识

2022-07-15,,,,

正文

接上章节:parsehtml 函数源码解析ast 基本形成

在正式扎进vue parse源码之前,我们先了解下他周边的工具函数, 这能帮我们快速的去理解阅读。

还记得我们在上章节讲的element元素节点的描述对象吗?

var element = {
	type: 1,
	tag: tag,
	parent: null,
	attrslist: attrs,
	children: []
}

在源码中定义了一个createastelement函数,用来创建一个元素的描述对象。

createastelement函数

function createastelement(tag, attrs, parent) {
		return {
			type: 1,
			tag: tag,
			attrslist: attrs,
			attrsmap: makeattrsmap(attrs),
			parent: parent,
			children: []
		}
	}

解析指令所用正则

var onre = /^@|^v-on:/;
var dirre = /^v-|^@|^:/;
var foraliasre = /([\s\s]*?)\s+(?:in|of)\s+([\s\s]*)/;
var foriteratorre = /,([^,\}\]]*)(?:,([^,\}\]]*))?$/;
var stripparensre = /^\(|\)$/g;
var argre = /:(.*)$/;
var bindre = /^:|^v-bind:/;
var modifierre = /\.[^.]+/g;

在解析开始标签的时候你遇到的不仅有属性,还有一些vue 自定义的指令。下面一起来分析下解析指令会有用哪些正则。

onre

var onre = /^@|^v-on:/;

匹配以字符 @ 或 v-on: 开头的字符串,主要作用是检测标签属性名是否是监听事件的指令。

dirre

var const dirre = /^v-|^@|^:/

匹配以字符 v- 或 @ 或 : 开头的字符串,主要作用是检测标签属性名是否是指令。在vue中所有以 v- 开头的属性都被认为是指令,另外@字符是 v-on 的缩写,: 字符是 v-bind 的缩写。

foraliasre

var foraliasre = /([\s\s]*?)\s+(?:in|of)\s+([\s\s]*)/;

匹配 v-for 属性的值,并捕获 in 或 of 前后的字符串。都是正则大神就不解释怎么捕获的了。

foriteratorre

var foriteratorre = /,([^,\}\]]*)(?:,([^,\}\]]*))?$/;

这个也是匹配v-for的属性值,不过比之前要稍微复杂点:列表渲染 v-for(https://cn.vuejs.org/v2/guide/list.html)需要先了解下这个。

//示例:1
<div v-for="(value, name) in object">
  {{ name }}: {{ value }}
</div>
//示例:2 
<div v-for="(value, name, index) in object">
  {{ index }}. {{ name }}: {{ value }}
</div>

没错就是用来捕获,示例1中的:'obj , index' 示例2中的:'value, key, index' 。

stripparensre

var stripparensre = /^\(|\)$/g;

这个捕获组用来捕获要么以字符 ( 开头,要么以字符 ) 结尾的字符串,或者两者都满足。那么这个正则的作用是什么呢?我们在讲解正则 foriteratorre 时有个细节不知道大家注意到了没有,就是 foriteratorre 正则所匹配的字符串是 'obj, index' ,而不是 '(obj, index)' ,这两个字符串的区别就在于第二个字符串拥有左右括号,所以在使用 foriteratorre 正则之前,需要使用 stripparensre 正则去掉字符串 '(obj, index)' 中的左右括号,实现方式很简单:

"(obj, index)".replace(stripparensre, "")

argre

var argre = /:(.*)$/;

argre正则用来匹配指令编写中的参数,并且拥有一个捕获组,用来捕获参数的名字。

示例:

<div v-on:click.item="handle"></div>

其中 v-on 为指令,click为传递给 v-on 指令的参数,stop 为修饰符。

bindre

var bindre = /^:|^v-bind:/;

该正则用来匹配以字符:或字符串 v-bind: 开头的字符串,主要用来检测一个标签的属性是否是绑定(v-bind)。

modifierre

var modifierre = /\.[^.]+/g;

该正则用来匹配修饰符的,但是并没有捕获任何东西,但你可以用match、exec等方法获取与当前正则匹配成功的信息。

parse 函数中的变量

在讲解 parse 函数直接我们还需要先了解下它内部所定义的一些变量以及用途。

function parse(template, options) {
	warn$2 = options.warn || basewarn;
	platformispretag = options.ispretag || no;
	platformmustuseprop = options.mustuseprop || no;
	platformgettagnamespace = options.gettagnamespace || no;
	transforms = pluckmodulefunction(options.modules, 'transformnode');
	pretransforms = pluckmodulefunction(options.modules, 'pretransformnode');
	posttransforms = pluckmodulefunction(options.modules, 'posttransformnode');
	delimiters = options.delimiters;
	var stack = [];
	var preservewhitespace = options.preservewhitespace !== false;
	var root;
	var currentparent;
	var invpre = false;
	var inpre = false;
	var warned = false;
	function warnonce(msg) {
        //...
	}
	function closeelement(element) {
       //...
	}
	parsehtml(template, {
		warn: warn$2,
		expecthtml: options.expecthtml,
		isunarytag: options.isunarytag,
		canbeleftopentag: options.canbeleftopentag,
		shoulddecodenewlines: options.shoulddecodenewlines,
		shoulddecodenewlinesforhref: options.shoulddecodenewlinesforhref,
		shouldkeepcomment: options.comments,
		start: function start(tag, attrs, unary) {},
		end: function end() {},
		chars: function chars(text) {},
		comment: function comment(text) {},
	});
	return root
}

我们先来看下针对web平台初始化的一些变量。

warn$2 = options.warn || basewarn;
platformispretag = options.ispretag || no;
platformmustuseprop = options.mustuseprop || no;
platformgettagnamespace = options.gettagnamespace || no;
transforms = pluckmodulefunction(options.modules, 'transformnode');
pretransforms = pluckmodulefunction(options.modules, 'pretransformnode');
posttransforms = pluckmodulefunction(options.modules, 'posttransformnode');
delimiters = options.delimiters;
  • warn$2 函数 毋庸置疑它作用是用来打印警告信息的
  • platformispretag 函数是一个编译器选项,其作用是通过给定的标签名字判断该标签是否是 pre 标签。
  • platformmustuseprop 该函数也是一个编译器选项,其作用是用来检测一个属性在标签中是否要使用元素对象原生的 prop 进行绑定。
  • platformgettagnamespace 该函数是一个编译器选项,其作用是用来获取元素(标签)的命名空间。
  • transforms 、pretransforms 、posttransforms 还没讲到它们的上下文,暂时不解释它们的作用。
  • delimiters 它的值为 options.delimiters 属性,它的值就是在创建 vue 实例对象时所传递的 delimiters 选项。

继续往下看:

var stack = [];
var preservewhitespace = options.preservewhitespace !== false;
var root;
var currentparent;
var invpre = false;
var inpre = false;
var warned = false;
  • stack 初始值是一个空数组,作用在上个章节我们讲到,回退操作为了让子元素描述对象的parent属性能够正确指向其父元素。
  • preservewhitespace 是一个布尔值并且它的值与编译器选项中的options.preservewhitespace选项有关,只要 options.preservewhitespace 的值不为false,那么 preservewhitespace 的值就为真。其中 options.preservewhitespace 选项用来告诉编译器在编译 html 字符串时是否放弃标签之间的空格,如果为 true 则代表放弃。
  • root 存储最终生成的ast。
  • currentparent 通过上章节了解到,该变量维护元素描述对象之间的父子关系。
  • invpre 初始值:false。标识当前解析的标签是否在拥有 v-pre (跳过这个元素和它的子元素的编译过程。)的标签之内。
  • inpre 初始值:false。标识当前正在解析的标签是否在 <pre></pre> 标签之内。
  • warned 初始值:false。用来打印警告信息的函数,只不过 warnonce 函数就如它的名字一样,只会打印一次警告信息,并且 warnonce 函数也是通过调用 warn 函数来实现的。

好了一些边边角角的东西就先讲到这,接下来我们一起来分析vue parse源码看看一颗完整的ast树是如何构建出来的。更多关于vue parsehtml函数ast预备的资料请关注其它相关文章!

《vue parseHTML函数源码解析 AST预备知识.doc》

下载本文的Word格式文档,以方便收藏与打印。