Vue的底層原理你了解多少
Observer (數(shù)據(jù)劫持)
核心是通過Obeject.defineProperty()來監(jiān)聽數(shù)據(jù)的變動,這個函數(shù)內(nèi)部可以定義setter和getter。
每當(dāng)數(shù)據(jù)發(fā)生變化,就會觸發(fā)setter()。這時候 Observer 就要通過 Dep 通知 Watcher 訂閱者。
Dep (發(fā)布者)
有 addWatcher() 和 notify() 兩個方法,(收集 Watcher 依賴,并通知依賴變更)。
Dep 保存多個 atcher,當(dāng) Dep 發(fā)現(xiàn) Observer 有更新時,Dep 會調(diào)用 notify() 方法去通知 Watcher 進行更新
Watcher (訂閱者)
有一個 update() 方法,(訂閱 Dep ,接收數(shù)據(jù)變更)。
Watcher 是 Observer 和 Compile 之間通信的橋梁,主要做的事情是:
1. 在自身實例化時往 Dep 中添加自己;
2. 待屬性變動接收到 Dep.notice() 通知時,能調(diào)用自身的 update() 方法,并觸發(fā)Compile中綁定的回調(diào)。
Compile
Compile主要做的事情是解析模板指令,將模板中的變量替換成數(shù)據(jù),然后初始化渲染頁面視圖,并將每個指令對應(yīng)的節(jié)點綁定更新函數(shù),添加監(jiān)聽數(shù)據(jù)的訂閱者,一旦接收到數(shù)據(jù)有變動,收到通知,更新視圖。
總結(jié)歸納:
vue 作為一種MVVM模式的框架, 其數(shù)據(jù)綁定的底層原理為:數(shù)據(jù)劫持 + 發(fā)布訂閱者模式。
其中主要有這么四種“角色”:
- Observer
- Dep數(shù)據(jù)收集
- Watcer訂閱者
- Compiler 模板編譯器。
Observer 主要負責(zé) 數(shù)據(jù)劫持, 核心是通過Obeject.defineProperty()來監(jiān)聽數(shù)據(jù)的變動,這個函數(shù)內(nèi)部可以定義setter和getter。每當(dāng)數(shù)據(jù)發(fā)生變化,就會觸發(fā)setter()。這時候 Observer 就要通知給Dep 說有數(shù)據(jù)發(fā)生了變化。
發(fā)布訂閱模式主要是通過 Dep 和 Watcher 來完成。
Dep 中存放著 Watcher 實例化時存放的所有依賴,是個數(shù)據(jù)集,當(dāng) Dep 收到來自 Observer 的數(shù)據(jù)變化通知時,會調(diào)用 notice() 方法把發(fā)生變化的依賴告訴 Watcher。
Watcher 是訂閱者,是連接 Observer 和 Compile 之間通信的橋梁,當(dāng)它收到來自 Dep 的數(shù)據(jù)變化通知后,會調(diào)用自身的 update() 方法,并觸發(fā)Compile中綁定的回調(diào)。
Compile 主要做的事情是解析模板指令,將模板中的變量替換成數(shù)據(jù),然后初始化渲染頁面視圖,并將每個指令對應(yīng)的節(jié)點綁定更新函數(shù),添加監(jiān)聽數(shù)據(jù)的訂閱者,一旦接收到數(shù)據(jù)有變動,收到通知,更新視圖。
源碼目錄結(jié)構(gòu)
├─ .circleci // 包含CircleCI持續(xù)集成/持續(xù)部署工具的配置文件 ├─ .github // 項目相關(guān)的說明文檔,上面的說明文檔就在此文件夾 ├─ benchmarks // 基準(zhǔn),性能測試文件,Vue的跑分demo,比如大數(shù)據(jù)量的table或者渲染大量SVG ├─ dist // 構(gòu)建后輸出的不同版本Vue文件(UMD、CommonJS、ES 生產(chǎn)和開發(fā)包) ├─ examples // 部分示例,用Vue寫的一些小demo ├─ flow // flow 因為Vue使用了 [Flow](https://flow.org/) 來進行靜態(tài)類型檢查,靜態(tài)類型檢查類型聲明文件 ├─ packages // 包含服務(wù)端渲染和模板編譯器兩種不同的NPM包,是提供給不同使用場景使用的 ├─ scripts // 存放npm腳本配置文件,結(jié)合webpack、rollup進行編譯、測試、構(gòu)建等操作(使用者不需要關(guān)心) │ ├─ alias.js // 模塊導(dǎo)入所有源代碼和測試中使用的別名 │ ├─ config.js // 包含在'dist/`中找到的所有文件的生成配置 │ ├─ build.js // 對 config.js 中所有的rollup配置進行構(gòu)建 ├─ src // 主要源碼所在位置,核心內(nèi)容 │ ├─ compiler // 解析模版相關(guān) │ ├─ codegen // 把AST轉(zhuǎn)換為Render函數(shù) │ ├─ directives // 通用生成Render函數(shù)之前需要處理的指令 │ ├─ parser // 解析模版成AST │ ├─ core // Vue核心代碼,包括內(nèi)置組件,全局API封裝,Vue 實例化,觀察者,虛擬DOM, 工具函數(shù)等等。 │ ├─ components // 組件相關(guān)屬性,主要是Keep-Alive │ ├─ global-api // Vue全局API,如Vue.use,Vue.extend,Vue.mixin等 │ ├─ instance // 實例化相關(guān)內(nèi)容,生命周期、事件等 │ ├─ observer // 響應(yīng)式核心目錄,雙向數(shù)據(jù)綁定相關(guān)文件 │ ├─ util // 工具方法 │ └─ vdom // 包含虛擬DOM 創(chuàng)建(creation)和打補丁(patching) 的代碼 │ ├─ platforms // 和平臺相關(guān)的內(nèi)容,Vue.js 是一個跨平臺的MVVM 框架(web、native、weex) │ ├─ web // web端 │ ├─ compiler // web端編譯相關(guān)代碼,用來編譯模版成render函數(shù)basic.js │ ├─ runtime // web端運行時相關(guān)代碼,用于創(chuàng)建Vue實例等 │ ├─ server // 服務(wù)端渲染 │ └─ util // 相關(guān)工具類 │ └─ weex // 基于通用跨平臺的 Web 開發(fā)語言和開發(fā)經(jīng)驗,來構(gòu)建 Android、iOS 和 Web 應(yīng)用 │ ├─ server // 服務(wù)端渲染(ssr) │ ├─ sfc // 轉(zhuǎn)換單文件組件(*.vue) │ └─ shared // 全局共享的方法和常量 ├─ test // test 測試用例 ├─ types // Vue新版本支持TypeScript,主要是TypeScript類型聲明文件 ├─ node_modules // npm包存放目錄 |-- .babelrc.js // babel配置 |-- .editorconfig // 文本編碼樣式配置文件 |-- .eslintignore // eslint校驗忽略文件 |-- .eslintrc.js // eslint配置文件 |-- .flowconfig // flow配置文件 |-- .gitignore // Git提交忽略文件配置 |-- BACKERS.md // 贊助者信息文件 |-- LICENSE // 項目開源協(xié)議 |-- package.json // 依賴 |-- README.md // 說明文件 |-- yarn.lock // yarn版本鎖定文件
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
vue?中使用?this?更新數(shù)據(jù)的一次問題記錄
這篇文章主要介紹了vue?中使用?this?更新數(shù)據(jù)的一次大坑?,我在 vue 實例中聲明了一個數(shù)組屬性如?books: [],在異步請求的回調(diào)函數(shù)中使用?this.books = res.data.data;?進行數(shù)據(jù)更新,感興趣的朋友跟隨小編一起看看吧2022-11-11ElementUI實現(xiàn)el-form表單重置功能按鈕
本文主要介紹了Element使用el-form時,點擊重置按鈕或者取消按鈕時會實現(xiàn)表單重置效果,具有一定的參考價值,感興趣的可以了解一下2021-07-07在vs code 中如何創(chuàng)建一個自己的 Vue 模板代碼
這篇文章主要介紹了在vs code 中如何創(chuàng)建一個自己的 Vue 模板代碼,本文通過圖文并茂的形式給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-11-11Vue中傳遞自定義參數(shù)到后端、后端獲取數(shù)據(jù)并使用Map接收參數(shù)
有些傳遞的參數(shù)是直接拼接到URL地址欄中的、但是為了統(tǒng)一管理、不能將傳遞的參數(shù)直接拼接到地址欄中,接下來通過本文給大家介紹Vue中傳遞自定義參數(shù)到后端、后端獲取數(shù)據(jù)并使用Map接收參數(shù),感興趣的朋友一起看看吧2022-10-10VUE3刷新頁面報錯問題解決:Uncaught?SyntaxError:Unexpected?token?&apo
這篇文章主要介紹了VUE3刷新頁面報錯:Uncaught?SyntaxError:?Unexpected?token?‘<‘,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-03-03