Vue狀態(tài)機(jī)的開(kāi)啟與停止操作詳細(xì)講解
上節(jié)討論了遞歸下降算法,但是狀態(tài)機(jī)何時(shí)停止沒(méi)有介紹,也就是isEnd()函數(shù)的判斷邏輯,為了搞清楚這個(gè)問(wèn)題,我們需要模擬狀態(tài)機(jī)的運(yùn)行過(guò)程。
前面介紹了,在調(diào)用parseElement函數(shù)解析標(biāo)簽節(jié)點(diǎn)時(shí),會(huì)遞歸地調(diào)用parseChildren函數(shù),從而開(kāi)啟新的狀態(tài)機(jī)。在狀態(tài)機(jī)都執(zhí)行完成后,這時(shí)父級(jí)節(jié)點(diǎn)棧為空,狀態(tài)機(jī)全部停止運(yùn)行,模板解析完畢
狀態(tài)機(jī)會(huì)遭遇不符合預(yù)期的狀態(tài),可以通過(guò)下面parseChildren函數(shù)的代碼來(lái)提現(xiàn)這一點(diǎn):
function parseChildren(context, ancestors){ let nodes = [] const { mode } = context while(!isEnd(context, ancestor)){ let node if(mode === TextModes.DATA || mode === TextModes.RCDATA){ if(mode===TextModes.DATA && context.source[0] === '<'){ if(context.source[1] === '!'){ // 省略部分代碼 }else if(context.source[1] === '/'){ // 狀態(tài)機(jī)遭遇了閉合標(biāo)簽,此時(shí)應(yīng)該拋出錯(cuò)誤,因?yàn)樗鄙倥c之對(duì)應(yīng)的開(kāi)始標(biāo)簽 console.error('無(wú)效的結(jié)束標(biāo)簽') continue }else if(/[a-z]/i,test(context.source[1])){ // 省略部分代碼 } } else if (context.source.startswith('{{')) [ // 省略部分代碼 } } // 省略部分代碼 } return nodes }
換句話說(shuō),按照我們當(dāng)前的實(shí)現(xiàn)思路來(lái)解析上述例子中的模板,最終得到的錯(cuò)誤信息是:“無(wú)效的結(jié)束標(biāo)簽”。
但其實(shí)還有另外一種更好的解析方式。觀察上例中給出的模板,其中存在一段完整的內(nèi)容,如下:
<div><span></div></span>
可以看到模板中存在一段完整的內(nèi)容我們希望解析器可以正常對(duì)其進(jìn)行解析,這很可能也是符合用戶意圖的。
但實(shí)際上,無(wú)論哪一種解釋方式,對(duì)程序的影響都不大。兩者的區(qū)別體現(xiàn)在錯(cuò)誤處理上。對(duì)于第一種解釋方式,我們得到的錯(cuò)誤信息是“無(wú)效的結(jié)束標(biāo)簽”。而對(duì)于第二種解釋方式,在“完整的內(nèi)容”部分被解析完畢后,解析器就會(huì)打印錯(cuò)誤信息:“<span>
標(biāo)簽缺少閉合標(biāo)簽”。很顯然,第二種解釋方式更加合理
為了實(shí)現(xiàn)第二種解釋方式我們需要調(diào)整 isEnd 函數(shù)的邏輯。當(dāng)判斷狀態(tài)機(jī)是否應(yīng)該停止時(shí),不應(yīng)該總是與棧頂?shù)母讣?jí)節(jié)點(diǎn)做比較,而是應(yīng)該與整個(gè)父級(jí)節(jié)點(diǎn)棧中的所有節(jié)點(diǎn)做比較。只要父級(jí)節(jié)點(diǎn)棧中存在于當(dāng)前遇到的結(jié)束標(biāo)簽同名的節(jié)點(diǎn),就停止?fàn)顟B(tài)機(jī),如下面的代碼所示:
function isEnd(context, ancestors){ if(!context.source) return true // 與父級(jí)節(jié)點(diǎn)棧內(nèi)所有節(jié)點(diǎn)做比較 for(let i = ancestors.length-1;i>=0;--i){ //只要棧中存在與當(dāng)前結(jié)束標(biāo)簽同名的節(jié)點(diǎn),就停止?fàn)顟B(tài)機(jī) if(context.source.startsWith(`</${ancestors[i].tag}`)){ return true } } }
按照新思路對(duì)下面的模板執(zhí)行解析:
<div><span></div></span>
其流程如下:
- “狀態(tài)機(jī)1”遇到
<div>
開(kāi)始標(biāo)簽,調(diào)用parseElement解析函數(shù),并開(kāi)啟“狀態(tài)機(jī)2”解析子節(jié)點(diǎn)。 - “狀態(tài)機(jī)2”遇到
<span>
開(kāi)始簽,調(diào)用 parseElement 解析函數(shù),并開(kāi)啟“狀態(tài)機(jī)3”解析子節(jié)點(diǎn)。 - “狀態(tài)機(jī)3”遇到
</div>
結(jié)束標(biāo)簽,由于節(jié)點(diǎn)棧中存在名為 div 的標(biāo)簽節(jié)點(diǎn),于是“狀態(tài)機(jī)3”停止了。
在這個(gè)過(guò)程中,“狀態(tài)機(jī)2”在調(diào)用 parseElement 解函數(shù)時(shí),parseElement 函數(shù)能夠發(fā)現(xiàn)<span>
缺少閉合標(biāo)簽,于是會(huì)打印錯(cuò)誤信息“<span>
標(biāo)簽缺少閉合標(biāo)簽”,如下面的代碼所示:
function parseElement(context, ancestors){ const element = parseTag(context) if(element.isSelfClosing) return element ancestors.push(element) element.children = parseChildren(context, ancestors) ancestors.pop() if(context.source.startsWith(`</${element.tag}`)){ parseTag(context, 'end') }else{ console.error(`${element.tag}標(biāo)簽缺少閉合標(biāo)簽`) } return element }
到此這篇關(guān)于Vue狀態(tài)機(jī)的開(kāi)啟與停止操作詳細(xì)講解的文章就介紹到這了,更多相關(guān)Vue狀態(tài)機(jī)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
HTML頁(yè)面中使用Vue示例進(jìn)階(快速學(xué)會(huì)上手Vue)
Vue是用于構(gòu)建用戶界面的漸進(jìn)式JavaScript框架。特色:構(gòu)建用戶界面—數(shù)據(jù)變成界面;漸進(jìn)式—Vue可以自底向上逐層的應(yīng)用。VUE有兩種使用方式,一種實(shí)在html中直接使用vue做開(kāi)發(fā),一種是企業(yè)級(jí)的單頁(yè)面應(yīng)用。2023-02-02Vue頁(yè)面中實(shí)現(xiàn)平滑滾動(dòng)功能
這是一個(gè)實(shí)現(xiàn)平滑滾動(dòng)的函數(shù),可以讓頁(yè)面在滾動(dòng)到指定位置時(shí)產(chǎn)生緩動(dòng)效果,本文給大家介紹了如何在在Vue頁(yè)面中實(shí)現(xiàn)平滑滾動(dòng)功能,<BR>,文中詳細(xì)的代碼講解供大家參考,具有一定的參考價(jià)值,需要的朋友可以參考下2023-12-12Vue3監(jiān)聽(tīng)reactive對(duì)象中屬性變化的方法
在 Vue 3 中,如果你想監(jiān)聽(tīng) reactive 對(duì)象中的某個(gè)屬性發(fā)生的變化,你可以使用 watch 函數(shù)進(jìn)行監(jiān)聽(tīng),watch 函數(shù)允許你觀察 reactive 對(duì)象的某個(gè)屬性或者整個(gè)對(duì)象,所以本文給大家介紹了Vue3監(jiān)聽(tīng)reactive對(duì)象中屬性變化的方法,需要的朋友可以參考下2024-08-08使用Vue3+PDF.js實(shí)現(xiàn)PDF預(yù)覽功能
項(xiàng)目中有一個(gè)需要預(yù)覽下載pdf的需求,網(wǎng)上找了很久,決定使用 pdf.js 完成,下面這篇文章主要給大家介紹了關(guān)于使用Vue3+PDF.js實(shí)現(xiàn)PDF預(yù)覽功能的相關(guān)資料,需要的朋友可以參考下2022-12-12Vue 頁(yè)面權(quán)限控制和登陸驗(yàn)證功能的實(shí)例代碼
這篇文章主要介紹了Vue 頁(yè)面權(quán)限控制和登陸驗(yàn)證功能的實(shí)例代碼,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-06-06