淺談vue中.vue文件解析流程
我們平時(shí)寫的 .vue 文件稱為 SFC(Single File Components),本文介紹將 SFC 解析為 descriptor 這一過程在 vue 中是如何執(zhí)行的。
vue 提供了一個(gè) compiler.parseComponent(file, [options]) 方法,來將 .vue 文件解析成一個(gè) descriptor:
// an object format describing a single-file component.
declare type SFCDescriptor = {
template: ?SFCBlock;
script: ?SFCBlock;
styles: Array<SFCBlock>;
customBlocks: Array<SFCBlock>;
};
文件入口
解析 sfc 文件的入口在 src/sfc/parser.js 中,該文件 export 了 parseComponent 方法, parseComponent 方法用來對(duì)單文件組件進(jìn)行編譯。
接下來我們看看 parseComponent 方法都做了哪些事情。
parseComponent 方法
function start(tag, attrs, unary, start, end,){
}
function end(tag, start, end){
}
parseHTML(content, {
start,
end
})
parseComponent 方法中定義了 start``end 兩個(gè)函數(shù),之后調(diào)用了 parseHTML 方法來對(duì) .vue 文件內(nèi)容踐行編譯。
那么這個(gè) parseHTML 方法是做啥的呢?
parseHTML 方法
該方法看名字就知道是一個(gè) html-parser,可以簡(jiǎn)單理解為,解析到每個(gè)起始標(biāo)簽時(shí),調(diào)用 option 中的 start;每個(gè)標(biāo)簽結(jié)束時(shí),調(diào)用 option 中的 end。
對(duì)應(yīng)到這里,就是分別調(diào)用 parseComponent 方法中定義的 start 和 end 函數(shù)。
在 parseComponent 中維護(hù)一個(gè) depth 變量,在 start 中將 depth++ ,在 end 中 depth-- 。那么,每個(gè) depth === 0 的標(biāo)簽就是我們需要獲取的信息,包含 template、script、style 以及一些自定義標(biāo)簽。
start
每當(dāng)遇到一個(gè)起始標(biāo)簽時(shí),執(zhí)行 start 函數(shù)。
1、記錄下 currentBlock。
每個(gè) currentBlock 包含以下內(nèi)容:
declare type SFCBlock = {
type: string;
content: string;
start?: number;
end?: number;
lang?: string;
src?: string;
scoped?: boolean;
module?: string | boolean;
};
2、根據(jù) tag 名稱,將 currentBlock 對(duì)象在返回結(jié)果對(duì)象中。
返回結(jié)果對(duì)象定義為 sfc,如果tag不是 script,style,template 中的任一個(gè),就放在 sfc.customBlocks 中。如果是style,就放在 sfc.styles 中。script 和 template 則直接放在 sfc 下。
if (isSpecialTag(tag)) {
checkAttrs(currentBlock, attrs)
if (tag === 'style') {
sfc.styles.push(currentBlock)
} else {
sfc[tag] = currentBlock
}
} else { // custom blocks
sfc.customBlocks.push(currentBlock)
}
end
每當(dāng)遇到一個(gè)結(jié)束標(biāo)簽時(shí),執(zhí)行 end 函數(shù)。
1、如果當(dāng)前是第一層標(biāo)簽(depth === 1),并且 currentBlock 變量存在,那么取出這部分text,放在 currentBlock.content 中。
if (depth === 1 && currentBlock) {
currentBlock.end = start
let text = deindent(content.slice(currentBlock.start, currentBlock.end))
// pad content so that linters and pre-processors can output correct
// line numbers in errors and warnings
if (currentBlock.type !== 'template' && options.pad) {
text = padContent(currentBlock, options.pad) + text
}
currentBlock.content = text
currentBlock = null
}
2、depth-- 。
得到 descriptor
在將 .vue 整個(gè)遍歷一遍后,得到的 sfc 對(duì)象即為我們需要的結(jié)果。
生成 .js ?
compiler.parseComponent(file, [options]) 得到的只是一個(gè)組件的 SFCDescriptor ,最終編譯成.js 文件是交給 vue-loader 等庫(kù)來做的。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
vue中如何動(dòng)態(tài)綁定圖片,vue中通過data返回圖片路徑的方法
下面小編就為大家分享一篇vue中如何動(dòng)態(tài)綁定圖片,vue中通過data返回圖片路徑的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-02-02
Vue2中pinia刷新后數(shù)據(jù)丟失的問題解決
Pinia是一個(gè)Vue.js狀態(tài)管理庫(kù),如果你在組件中修改了store中的數(shù)據(jù)并刷新了界面,Pinia會(huì)將store中的數(shù)據(jù)重置為初始值,從而導(dǎo)致數(shù)據(jù)丟失的問題,本文就來介紹一下問題解決,感興趣的可以了解一下2023-12-12
Vue3使用src動(dòng)態(tài)引入本地圖片的詳細(xì)步驟
這篇文章主要給大家介紹了關(guān)于Vue3使用src動(dòng)態(tài)引入本地圖片的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-12-12
Vue項(xiàng)目中components組件(模板)的使用及說明
這篇文章主要介紹了Vue項(xiàng)目中components組件(模板)的使用及說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05
基于Vue實(shí)現(xiàn)簡(jiǎn)單的貪食蛇游戲
貪食蛇是一個(gè)非常經(jīng)典的游戲,?在游戲中,?玩家操控一條細(xì)長(zhǎng)的直線,?它會(huì)不停前進(jìn),?玩家只能操控蛇的頭部朝向,?一路拾起觸碰到之物。本文將用Vue實(shí)現(xiàn)這一游戲,感興趣的可以嘗試一下2022-04-04
vue-router之路由鉤子函數(shù)應(yīng)用小結(jié)
vue-router提供的導(dǎo)航鉤子主要用來攔截導(dǎo)航,讓它完成跳轉(zhuǎn)或取消,本文主要介紹了vue-router之路由鉤子函數(shù)應(yīng)用小結(jié),具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01
vue使用高德地圖點(diǎn)擊下鉆上浮效果的實(shí)現(xiàn)思路
這篇文章主要介紹了vue使用高德地圖點(diǎn)擊下鉆 上浮效果的實(shí)現(xiàn)思路,本文以浙江省為例通過實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2019-10-10

