首頁 > 軟體

Vue的底層原理你瞭解多少

2022-02-14 16:00:25

Observer (資料劫持)

核心是通過Obeject.defineProperty()來監聽資料的變動,這個函數內部可以定義setter和getter。
每當資料發生變化,就會觸發setter()。這時候 Observer 就要通過 Dep 通知 Watcher 訂閱者。

Dep (釋出者)

有 addWatcher() 和 notify() 兩個方法,(收集 Watcher 依賴,並通知依賴變更)

Dep 儲存多個 atcher,當 Dep 發現 Observer 有更新時,Dep 會呼叫 notify() 方法去通知 Watcher 進行更新

Watcher (訂閱者)

有一個 update() 方法,(訂閱 Dep ,接收資料變更)

Watcher 是 Observer 和 Compile 之間通訊的橋樑,主要做的事情是:

1. 在自身範例化時往 Dep 中新增自己;

2. 待屬性變動接收到 Dep.notice() 通知時,能呼叫自身的 update() 方法,並觸發Compile中繫結的回撥。

Compile

Compile主要做的事情是解析模板指令,將模板中的變數替換成資料,然後初始化渲染頁面檢視,並將每個指令對應的節點繫結更新函數,新增監聽資料的訂閱者,一旦接收到資料有變動,收到通知,更新檢視

總結歸納:

vue 作為一種MVVM模式的框架, 其資料繫結的底層原理為:資料劫持 + 釋出訂閱者模式

其中主要有這麼四種“角色”

  • Observer 
  • Dep資料收集
  • Watcer訂閱者
  • Compiler 模板編譯器。

Observer 主要負責 資料劫持, 核心是通過Obeject.defineProperty()來監聽資料的變動,這個函數內部可以定義setter和getter。每當資料發生變化,就會觸發setter()。這時候 Observer 就要通知給Dep 說有資料發生了變化。

釋出訂閱模式主要是通過 Dep 和 Watcher 來完成。

Dep 中存放著 Watcher 範例化時存放的所有依賴,是個資料集,當 Dep 收到來自 Observer 的資料變化通知時,會呼叫 notice() 方法把發生變化的依賴告訴 Watcher。

Watcher 是訂閱者,是連線 Observer 和 Compile 之間通訊的橋樑,當它收到來自 Dep 的資料變化通知後,會呼叫自身的 update() 方法,並觸發Compile中繫結的回撥。

Compile 主要做的事情是解析模板指令,將模板中的變數替換成資料,然後初始化渲染頁面檢視,並將每個指令對應的節點繫結更新函數,新增監聽資料的訂閱者,一旦接收到資料有變動,收到通知,更新檢視。

原始碼目錄結構

├─ .circleci                   // 包含CircleCI持續整合/持續部署工具的組態檔
├─ .github                   // 專案相關的說明檔案,上面的說明檔案就在此資料夾
├─ benchmarks                 // 基準,效能測試檔案,Vue的跑分demo,比如巨量資料量的table或者渲染大量SVG
├─ dist                       // 構建後輸出的不同版本Vue檔案(UMD、CommonJS、ES 生產和開發包)
├─ examples                   // 部分範例,用Vue寫的一些小demo
├─ flow                       // flow 因為Vue使用了 [Flow](https://flow.org/) 來進行靜態型別檢查,靜態型別檢查型別宣告檔案
├─ packages                   // 包含伺服器端渲染和模板編譯器兩種不同的NPM包,是提供給不同使用場景使用的
├─ scripts                   // 存放npm指令碼組態檔,結合webpack、rollup進行編譯、測試、構建等操作(使用者不需要關心)
│   ├─ alias.js              // 模組匯入所有原始碼和測試中使用的別名
│   ├─ config.js             // 包含在'dist/`中找到的所有檔案的生成設定
│   ├─ build.js               // 對 config.js 中所有的rollup設定進行構建
├─ src                        // 主要原始碼所在位置,核心內容
│   ├─ compiler               // 解析模版相關
│       ├─ codegen            // 把AST轉換為Render函數
│       ├─ directives         // 通用生成Render函數之前需要處理的指令
│       ├─ parser              // 解析模版成AST
│   ├─ core                    // Vue核心程式碼,包括內建元件,全域性API封裝,Vue 範例化,觀察者,虛擬DOM, 工具函數等等。
│       ├─ components          // 元件相關屬性,主要是Keep-Alive
│       ├─ global-api          // Vue全域性API,如Vue.use,Vue.extend,Vue.mixin等
│       ├─ instance            // 範例化相關內容,生命週期、事件等
│       ├─ observer            // 響應式核心目錄,雙向資料繫結相關檔案
│       ├─ util                // 工具方法
│       └─ vdom                // 包含虛擬DOM 建立(creation)和打修補程式(patching) 的程式碼
│   ├─ platforms               // 和平臺相關的內容,Vue.js 是一個跨平臺的MVVM 框架(web、native、weex)
│       ├─ web                 // web端
│           ├─ compiler        // web端編譯相關程式碼,用來編譯模版成render函數basic.js
│           ├─ runtime         // web端執行時相關程式碼,用於建立Vue範例等
│           ├─ server          // 伺服器端渲染
│           └─ util            // 相關工具類
│       └─ weex                // 基於通用跨平臺的 Web 開發語言和開發經驗,來構建 Android、iOS 和 Web 應用
│   ├─ server                  // 伺服器端渲染(ssr)
│   ├─ sfc                     // 轉換單檔案元件(*.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                 // 專案開源協定
|-- package.json             // 依賴
|-- README.md               // 說明檔案
|-- yarn.lock               // yarn版本鎖定檔案
 

總結

本篇文章就到這裡了,希望能夠給你帶來幫助,也希望您能夠多多關注it145.com的更多內容!  


IT145.com E-mail:sddin#qq.com