溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Vue編譯器分析compile源碼

發(fā)布時間:2022-07-14 09:47:03 來源:億速云 閱讀:147 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要介紹“Vue編譯器分析compile源碼”的相關(guān)知識,小編通過實(shí)際案例向大家展示操作過程,操作方法簡單快捷,實(shí)用性強(qiáng),希望這篇“Vue編譯器分析compile源碼”文章能幫助大家解決問題。

引言

在 compileToFunctions 方法中:

// compile
var compiled = compile(template, options);

而真正的編譯工作是依托于 compile 函數(shù),接下來我們詳細(xì)解析 compile 。

解析 compile

上述代碼在調(diào)用 compile ,其中模板字符串 template ,選項(xiàng)參數(shù) options 第二個參數(shù)傳遞給 compile 函數(shù),在章節(jié)三種我們知道,這里傳遞過去的options如下:

{
	shouldDecodeNewlines,
	shouldDecodeNewlinesForHref,
	delimiters,
	comments,
	warn 
}

compile 源碼

接下來我們看下 compile 的源碼。

function createCompilerCreator(baseCompile) {
	return function createCompiler(baseOptions) {
		function compile(
			template,
			options
		) {
			var finalOptions = Object.create(baseOptions);
			var errors = [];
			var tips = [];
			finalOptions.warn = function(msg, tip) {
				(tip ? tips : errors).push(msg);
			};
			if (options) {
				// merge custom modules
				if (options.modules) {
					finalOptions.modules =
						(baseOptions.modules || []).concat(options.modules);
				}
				// merge custom directives
				if (options.directives) {
					finalOptions.directives = extend(
						Object.create(baseOptions.directives || null),
						options.directives
					);
				}
				// copy other options
				for (var key in options) {
					if (key !== 'modules' && key !== 'directives') {
						finalOptions[key] = options[key];
					}
				}
			}
			var compiled = baseCompile(template, finalOptions); {
				errors.push.apply(errors, detectErrors(compiled.ast));
			}
			compiled.errors = errors;
			compiled.tips = tips;
			return compiled
		}
		return {
			compile: compile,
			compileToFunctions: createCompileToFunctionFn(compile)
		}
	}
}

·首先可以看到:

var finalOptions = Object.create(baseOptions);

配置選項(xiàng)

finalOptions 所有的配置選項(xiàng)最終都會掛載在這個對象上,baseOptions包含編譯器在運(yùn)作的時候所需的配置選項(xiàng)。

var baseOptions = {
	expectHTML: true,
	modules: modules$1,
	directives: directives$1,
	isPreTag: isPreTag,
	isUnaryTag: isUnaryTag,
	mustUseProp: mustUseProp,
	canBeLeftOpenTag: canBeLeftOpenTag,
	isReservedTag: isReservedTag,
	getTagNamespace: getTagNamespace,
	staticKeys: genStaticKeys(modules$1)
};

屬性分別解析

  • 第一個屬性: expectHTML 被設(shè)置為 true 。

  • 第二個屬性:modules

var modules$1 = [
	klass$1,
	style$1,
	model$1
];
var klass$1 = {
	staticKeys: ['staticClass'],
	transformNode: transformNode,
	genData: genData
};
var style$1 = {
	staticKeys: ['staticStyle'],
	transformNode: transformNode$1,
	genData: genData$1
};
var model$1 = {
	preTransformNode: preTransformNode
};

我們用到了在細(xì)講。

  • 第三個屬性:directives 值是三個屬性 (model、text、html) 的對象,且屬性的值都是函數(shù)。

  • 第四個屬性:isPreTag 它是一個函數(shù),其作用是通過給定的標(biāo)簽名字檢查標(biāo)簽是否是 'pre' 標(biāo)簽。

  • 第五個屬性:isUnaryTag 是一個通過makeMap生成的函數(shù),該函數(shù)的作用是檢測給定的標(biāo)簽是否是一元標(biāo)簽。

  • 第六個屬性:mustUseProp 它是一個函數(shù),其作用是用來檢測一個屬性在標(biāo)簽中是否要使用props進(jìn)行綁定。

  • 第七個屬性:canBeLeftOpenTag 一個使用makeMap生成的函數(shù),它的作用是檢測非一元標(biāo)簽,但卻可以自己補(bǔ)全并閉合的標(biāo)簽。比如 div 標(biāo)簽是一個雙標(biāo)簽,你需要這樣使用<div> text </div>,但是你依然可以省略閉合標(biāo)簽,直接這樣寫:<div> text ,且瀏覽器會自動補(bǔ)全。但是有些標(biāo)簽?zāi)悴豢梢赃@樣用,它們是嚴(yán)格的雙標(biāo)簽。

  • 第八個屬性:isReservedTag 它是一個函數(shù),其作用是檢查給定的標(biāo)簽是否是保留的標(biāo)簽。

  • 第九個屬性:getTagNamespace 它也是一個函數(shù),其作用是獲取元素(標(biāo)簽)的命名空間。

  • 第十個屬性:staticKeys 它的值是通過以 modules 為參數(shù)調(diào)用 genStaticKeys 函數(shù)的返回值得到的。 其作用是根據(jù)編譯器選項(xiàng)的 modules 選項(xiàng)生成一個靜態(tài)鍵字符串。

現(xiàn)在我們粗略的介紹了下baseOptions 中各個屬性的作用,當(dāng)我們用到時候再來詳細(xì)講解他們的源碼。

繼續(xù)往下看:

var errors = [];
var tips = [];
finalOptions.warn = function(msg, tip) {
	(tip ? tips : errors).push(msg);
};

很簡單

finalOptions添加warn 方法

在 finalOptions上添加了 warn 方法,該方法接收兩個參數(shù):

  • msg 錯誤或提示的信息

  • tip 用來標(biāo)示 msg 是錯誤還是提示。

warn選項(xiàng)主要用在編譯過程中的錯誤和提示收集,如果收集的信息是錯誤信息就將錯誤信息添加到前面定義的errors數(shù)組里,如果是提示信息就將其添加到 tips 數(shù)組里。

繼續(xù):

if (options) {
	// merge custom modules
	if (options.modules) {
		finalOptions.modules =
			(baseOptions.modules || []).concat(options.modules);
	}
	// merge custom directives
	if (options.directives) {
		finalOptions.directives = extend(
			Object.create(baseOptions.directives || null),
			options.directives
		);
	}
	// copy other options
	for (var key in options) {
		if (key !== 'modules' && key !== 'directives') {
			finalOptions[key] = options[key];
		}
	}
}

這段代碼檢查 options 是否存在,這里的 options 就是使用編譯器編譯模板時傳遞的選項(xiàng)參數(shù),或者可以簡單理解為調(diào)用 compileToFunctions 函數(shù)時傳遞的選項(xiàng)參數(shù)。

而baseOptions理解為編譯器的默認(rèn)選項(xiàng)或者基本選項(xiàng),options 是用來提供定制能力的擴(kuò)展選項(xiàng)。而上面這段代碼的作用,就是將 options 對象混合到 finalOptions 中。

兩個特殊的屬性處理

  • modules: 如果 options.modules 存在,就在 finalOptions 對象上添加 modules 屬性,其值為 baseOptions.modules 和 options.modules 這兩個數(shù)組合并后的新數(shù)組。

  • directives: 對于directives 采用原型鏈的原理實(shí)現(xiàn)擴(kuò)展屬性對基本屬性的覆蓋。

繼續(xù):

var compiled = baseCompile(template, finalOptions); {
	errors.push.apply(errors, detectErrors(compiled.ast));
}
compiled.errors = errors;
compiled.tips = tips;

上面的代碼調(diào)用了 baseCompile 函數(shù),并分別將字符串模板(template),以及最終的編譯器選項(xiàng)(finalOptions)傳遞了過去。

compiled 是 baseCompile 對模板的編譯結(jié)果所以上面這段代碼的作用是用來通過抽象語法樹來檢查模板中是否存在錯誤表達(dá)式的,通過 detectErrors 函數(shù)實(shí)現(xiàn),將compiled.ast 作為參數(shù)傳遞給 detectErrors 函數(shù),該函數(shù)最終返回一個數(shù)組,該數(shù)組中包含了所有錯誤的收集,最終通過這句代碼將錯誤添加到errors。

將收集到的錯誤(errors)和提示(tips)添加到compiled上并返回。

baseCompile 函數(shù)是在 createCompilerCreator 函數(shù)調(diào)用時傳遞的實(shí)參。

// `createCompilerCreator` allows creating compilers that use alternative
// parser/optimizer/codegen, e.g the SSR optimizing compiler.
// Here we just export a default compiler using the default parts.
var createCompiler = createCompilerCreator(function baseCompile(
	template,
	options
) {
	var ast = parse(template.trim(), options);
	if (options.optimize !== false) {
		optimize(ast, options);
	}
	var code = generate(ast, options);
	return {
		ast: ast,
		render: code.render,
		staticRenderFns: code.staticRenderFns
	}
});

關(guān)于“Vue編譯器分析compile源碼”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識,可以關(guān)注億速云行業(yè)資訊頻道,小編每天都會為大家更新不同的知識點(diǎn)。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI