摘要:實際上,我在看代碼的過程中順手提交了這個,作者眼明手快,當天就進行了修復,現(xiàn)在最新的代碼里已經(jīng)不是這個樣子了而且狀態(tài)機標識由字符串換成了數(shù)字常量,解析更準確的同時執(zhí)行效率也會更高。
最近饒有興致的又把最新版?Vue.js?的源碼學習了一下,覺得真心不錯,個人覺得 Vue.js 的代碼非常之優(yōu)雅而且精辟,作者本身可能無 (bu) 意 (xie) 提及這些。那么,就讓我來吧:)
程序結(jié)構(gòu)梳理Vue.js 是一個非常典型的 MVVM 的程序結(jié)構(gòu),整個程序從最上層大概分為
全局設(shè)計:包括全局接口、默認選項等
vm 實例設(shè)計:包括接口設(shè)計 (vm 原型)、實例初始化過程設(shè)計 (vm 構(gòu)造函數(shù))
這里面大部分內(nèi)容可以直接跟 Vue.js 的官方 API 參考文檔對應(yīng)起來,但文檔里面沒有且值得一提的是構(gòu)造函數(shù)的設(shè)計,下面是我摘出的構(gòu)造函數(shù)最核心的工作內(nèi)容。
整個實例初始化的過程中,重中之重就是把數(shù)據(jù) (Model) 和視圖 (View) 建立起關(guān)聯(lián)關(guān)系。Vue.js 和諸多 MVVM 的思路是類似的,主要做了三件事:
通過 observer 對 data 進行了監(jiān)聽,并且提供訂閱某個數(shù)據(jù)項的變化的能力
把 template 解析成一段 document fragment,然后解析其中的 directive,得到每一個 directive 所依賴的數(shù)據(jù)項及其更新方法。比如?v-text="message"?被解析之后 (這里僅作示意,實際程序邏輯會更嚴謹而復雜):
所依賴的數(shù)據(jù)項?this.$data.message,以及
相應(yīng)的視圖更新方法?node.textContent = this.$data.message
通過 watcher 把上述兩部分結(jié)合起來,即把 directive 中的數(shù)據(jù)依賴訂閱在對應(yīng)數(shù)據(jù)的 observer 上,這樣當數(shù)據(jù)變化的時候,就會觸發(fā) observer,進而觸發(fā)相關(guān)依賴對應(yīng)的視圖更新方法,最后達到模板原本的關(guān)聯(lián)效果。
所以整個 vm 的核心,就是如何實現(xiàn) observer, directive (parser), watcher 這三樣東西
文件結(jié)構(gòu)梳理Vue.js 源代碼都存放在項目的?src?目錄中,我們主要關(guān)注一下這個目錄 (事實上?test/unit/specs?目錄也值得一看,它是對應(yīng)著每個源文件的測試用例)。
src?目錄下有多個并列的文件夾,每個文件夾都是一部分獨立而完整的程序設(shè)計。不過在我看來,這些目錄之前也是有更立體的關(guān)系的:
首先是?api/*?目錄,這幾乎是最“上層”的接口封裝,實際的實現(xiàn)都埋在了其它文件夾里
然后是?instance/init.js,如果大家希望自頂向下了解所有 Vue.js 的工作原理的話,建議從這個文件開始看起
instance/scope.js:數(shù)據(jù)初始化,相關(guān)的子程序 (目錄) 有?observer/*、watcher.js、batcher.js,而?observer/dep.js?又是數(shù)據(jù)觀察和視圖依賴相關(guān)聯(lián)的關(guān)鍵
instance/compile.js:視圖初始化,相關(guān)的子程序 (目錄) 有?compiler/*、directive.js、parsers/*
其它核心要素:directives/*、element-directives/*、filters/*、transition/*
當然還有?util/*?目錄,工具方法集合,其實還有一個類似的?cache.js
最后是?config.js?默認配置項
篇幅有限,如果大家有意“通讀” Vue.js 的話,個人建議順著上面的整體介紹來閱讀賞析。
接下來是一些自己覺得值得一提的代碼細節(jié)
一些不容錯過的代碼/程序細節(jié) this._eventsCount?是什么?一開始看?instance/init.js?的時候,我立刻注意到一個細節(jié),就是?this._eventsCount = {}?這句,后面還有注釋
for $broadcast optimization
非常好奇,然后帶著疑問繼續(xù)看了下去,直到看到?api/events.js?中?$broadcast?方法的實現(xiàn),才知道這是為了避免不必要的深度遍歷:在有廣播事件到來時,如果當前 vm 的?_eventsCount?為?0,則不必向其子 vm 繼續(xù)傳播該事件。而且這個文件稍后也有?_eventsCount?計數(shù)的實現(xiàn)方式。
這是一種很巧妙同時也可以在很多地方運用的性能優(yōu)化方法。
數(shù)據(jù)更新的 diff 機制前陣子有很多關(guān)于視圖更新效率的討論,我猜主要是因為 virtual dom 這個概念的提出而導致的吧。這次我詳細看了一下 Vue.js 的相關(guān)實現(xiàn)原理。
實際上,視圖更新效率的焦點問題主要在于大列表的更新和深層數(shù)據(jù)更新這兩方面,而被熱烈討論的主要是前者 (后者是因為需求小還是沒爭議我就不得而知了)。所以這里著重介紹一下?directives/repeat.js?里對于列表更新的相關(guān)代碼。
首先?diff(data, oldVms)?這個函數(shù)的注釋對整個比對更新機制做了個簡要的闡述,大概意思是先比較新舊兩個列表的 vm 的數(shù)據(jù)的狀態(tài),然后差量更新 DOM。
第一步:遍歷新列表里的每一項,如果該項的 vm 之前就存在,則打一個?_reused?的標 (這個字段我一開始看?init.js?的時候也是困惑的…… 看到這里才明白意思),如果不存在對應(yīng)的 vm,則創(chuàng)建一個新的。
第二步:遍歷舊列表里的每一項,如果?_reused?的標沒有被打上,則說明新列表里已經(jīng)沒有它了,就地銷毀該 vm。
第三步:整理新的 vm 在視圖里的順序,同時還原之前打上的?_reused?標。就此列表更新完成。
順帶提一句 Vue.js 的元素過渡動畫處理 (v-transition) 也設(shè)計得非常巧妙,感興趣的自己看吧,就不展開介紹了
組件的?[keep-alive]?特性Vue.js 為其組件設(shè)計了一個?[keep-alive]?的特性,如果這個特性存在,那么在組件被重復創(chuàng)建的時候,會通過緩存機制快速創(chuàng)建組件,以提升視圖更新的性能。代碼在?directives/component.js。
數(shù)據(jù)監(jiān)聽機制如何監(jiān)聽某一個對象屬性的變化呢?我們很容易想到?Object.defineProperty?這個 API,為此屬性設(shè)計一個特殊的 getter/setter,然后在 setter 里觸發(fā)一個函數(shù),就可以達到監(jiān)聽的效果。
不過數(shù)組可能會有點麻煩,Vue.js 采取的是對幾乎每一個可能改變數(shù)據(jù)的方法進行 prototype 更改:
但這個策略主要面臨兩個問題:
無法監(jiān)聽數(shù)據(jù)的?length,導致?arr.length?這樣的數(shù)據(jù)改變無法被監(jiān)聽
通過角標更改數(shù)據(jù),即類似?arr[2] = 1?這樣的賦值操作,也無法被監(jiān)聽
為此 Vue.js 在文檔中明確提示不建議直接角標修改數(shù)據(jù)
同時 Vue.js 提供了兩個額外的“糖方法”?$set?和?$remove?來彌補這方面限制帶來的不便。整體上看這是個取舍有度的設(shè)計。我個人之前在設(shè)計數(shù)據(jù)綁定庫的時候也采取了類似的設(shè)計 (一個半途而廢的內(nèi)部項目就不具體獻丑了),所以比較認同也有共鳴。
path 解析器的狀態(tài)機設(shè)計首先要說?parsers?文件夾里有各種“財寶”等著大家挖掘!認真看一看一定不會后悔的
parsers/path.js?主要的職責是可以把一個 JSON 數(shù)據(jù)里的某一個“路徑”下的數(shù)據(jù)取出來,比如:
var path = "a.b[1].v" var obj = { a: { b: [ {v: 1}, {v: 2}, {v: 3} ] } } parse(obj, path) // 2
所以對?path?字符串的解析成為了它的關(guān)鍵。Vue.js 是通過狀態(tài)機管理來實現(xiàn)對路徑的解析的:
咋一看很頭大,不過如果再稍微梳理一下:
也許看得更清楚一點了,當然也能發(fā)現(xiàn)其中有一點小問題,就是源代碼中?inIdent?這個狀態(tài)是具有二義性的,它對應(yīng)到了圖中的三個地方,即?in ident?和兩個?in (quoted) ident。
實際上,我在看代碼的過程中順手提交了這個 bug,作者眼明手快,當天就進行了修復,現(xiàn)在最新的代碼里已經(jīng)不是這個樣子了:
而且狀態(tài)機標識由字符串換成了數(shù)字常量,解析更準確的同時執(zhí)行效率也會更高。
一點自己的思考首先是視圖的解析過程,Vue.js 的策略是把 element 或 template string 先統(tǒng)一轉(zhuǎn)換成 document fragment,然后再分解和解析其中的子組件和 directives。我覺得這里有一定的性能優(yōu)化空間,畢竟 DOM 操作相比之余純 JavaScript 運算還是會慢一些。
然后是基于移動端的思考,Vue.js 雖確實已經(jīng)非常非常小巧了 (min+gzip 之后約 22 kb),但它是否可以更小,繼續(xù)抽象出常用的核心功能,同時更快速,也是個值得思考的問題。
第三我非常喜歡通過 Vue.js 進行模塊化開發(fā)的模式,Vue 是否也可以借助類似 web components + virtual dom 的形態(tài)把這樣的開發(fā)模式帶到更多的領(lǐng)域,也是件很有意義的事情。
總結(jié)Vue.js 里的代碼細節(jié)還不僅于此,比如:
cache.js?里的緩存機制設(shè)計和場景運用 (如在?parsers/path.js?中)
parsers/template.js?里的?cloneNode?方法重寫和對 HTML 自動補全機制的兼容
在開發(fā)和生產(chǎn)環(huán)境分別用注釋結(jié)點和不可見文本結(jié)點作為視圖的“占位符”等等
自己也在閱讀代碼,了解 Vue.js 的同時學到了很多東西,同時我覺得代碼實現(xiàn)只是 Vue.js 優(yōu)秀的要素之一,整體的程序設(shè)計、API 設(shè)計、細節(jié)的取舍、項目的工程考量都非常棒!
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/40235.html
摘要:實際上,我在看代碼的過程中順手提交了這個,作者眼明手快,當天就進行了修復,現(xiàn)在最新的代碼里已經(jīng)不是這個樣子了而且狀態(tài)機標識由字符串換成了數(shù)字常量,解析更準確的同時執(zhí)行效率也會更高。 最近饒有興致的又把最新版?Vue.js?的源碼學習了一下,覺得真心不錯,個人覺得 Vue.js 的代碼非常之優(yōu)雅而且精辟,作者本身可能無 (bu) 意 (xie) 提及這些。那么,就讓我來吧:) 程序結(jié)構(gòu)梳...
摘要:有興趣的同學可以查看之前發(fā)布的文章學習系列一學習實踐筆記附學習系列二學習實踐筆記附學習系列三和網(wǎng)絡(luò)傳輸相關(guān)知識的學習實踐學習系列四打包工具的使用學習系列五從來聊聊學習系列項目地址項目暫時有點亂,之后會進行整理優(yōu)化。 上次學習了vue-router的使用,讓我能夠在各個頁面間切換,將頁面搭建了起來。這次則要學習vue的狀態(tài)管理模式——vuex。它類似于redux來應(yīng)用的全局狀態(tài)。 注:本...
閱讀 3079·2023-04-25 18:54
閱讀 2598·2021-11-02 14:40
閱讀 3193·2021-09-23 11:58
閱讀 2438·2019-08-30 13:50
閱讀 1243·2019-08-29 12:46
閱讀 3129·2019-08-28 17:51
閱讀 687·2019-08-26 11:47
閱讀 907·2019-08-23 16:17