主頁 > 企業開發 > 手牽手帶你實作mini-vue

手牽手帶你實作mini-vue

2023-06-20 09:22:08 企業開發

1 前言

隨著 Vue、React、Angularjs 等框架的誕生,資料驅動視圖的理念也深入人心,就 Vue 來說,它擁有著雙向資料系結、虛擬dom、組件化、視圖與資料相分離等等造福程式員的優點,那 Vue 的雙向資料系結實作原理是什么樣的,如果讓我們自己去實作一個這樣的雙向資料系結要怎么做呢,本文就與大家分享一下 Vue 的系結原理及其簡單實作

2 核心技術

大家都知道 Vue2 雙向系結是基于 ES5的 Object.defineProperty 方法+發布訂閱者模式實作的 那我們首先簡單了解一下這兩個模塊都是做什么的,在 Vue 中充當了什么角色

2.1 Object.defineProperty

用來在物件上定義或者修改一個屬性值,實作資料劫持,為修改資料后去呼叫視圖更新做準備

  const obj = {}
  let age = 18
  Object.defineProperty(obj, 'age',{
  get() {
      return age
   },
    set(newVal) {
     age = newVal + 1
    },
    enumerable: true
   })
 console.log(obj.age) // 18
 obj.age = 20
    console.log(obj.age) // 21

2.2 發布訂閱者模式

此模式簡單來講就是分為發布和訂閱兩個概念,訂閱意思就是我們會定義很多個訂閱者,每個訂閱者都會有自己的 update 方法,把需要更新的訂閱者放到陣列中,而發布就代表通知訂閱者去依次執行其 update 方法,從而實作資料更新

// 定義放訂閱者的陣列
function Dep() {
  this.subs = []
}
// 定義存放訂閱者的方法
Dep.prototype.addSub = function(sub) {
  this.subs.push(sub)
}
// 定義發布的方法
Dep.prototype.notify = function(sub) {
  this.subs.forEach(sub => {
    // 依次通知訂閱者去執行update方法
    sub.update()
  })
}

寫到這里我們就把發布和訂閱準備好了,但是還缺少訂閱者,且訂閱者要保證提供一個 update 方法才行,那我們不禁想到能否去創建一個建構式,通過這個建構式創建的實體都會有 update 方法呢

function Watcher(fn) {
  this.fn = fn
}
// 通過該建構式創建的實體都會有update方法
Watcher.prototype.update = function() {
  this.fn()
}
// new實體
const watcher1 = new Watcher(() => console.log('我是watcher1'))
const watcher2 = new Watcher(() => console.log('我是watcher2'))
const dep = new Dep()
// 把準備好的事件放入到陣列中
dep.addSub(watcher1)
dep.addSub(watcher2)
// 進行發布
dep.notify()
// 最終輸出 我是watcher1 我是watcher2

3 具體實作

3.1 初始化

一個框架都是從它的初始化開始的,Vue 也不例外

<body>
    <div id="app">
      <p>a value: {{a.a}}</p>
      <div>b value: {{b}}</div>
      <span>v-model: </span><input type="text" v-model="b">
    </div>
    <script>
      // 模仿 Vue 的初始化和傳入的引數
      let vue = new Vue({
        el: '#app',
        data: {
          a: { a: 'is a' },
          b: 'is b',
          c: 'is c'
        }
      })
    </script>
  </body>

3.2 資料劫持 observe

Vue 中在 data 中定義的屬性才可以實作雙向系結,為了實作這個功能,我們定義一個 Observe 用來劫持到物件的屬性

// 給物件增加資料劫持
function Observe(data) {
  // 因 defineProperty 每次只能設定單個屬性 所以需遍歷
  for (let key in data) {
    let val = data[key]
    observe(val)
    Object.defineProperty(data, key, {
      enumerable: true,
      get() {
        return val
      },
      set(newVal) {
        if (newVal === val) return // 新值與舊值相等時 不做處理
        val = newVal // 之所以給 val 賦值 是因為取值時取得val
        observe(newVal) // 當給變數值賦予一個新物件時 依然需要劫持到其屬性
      }
    })
  }
}
function observe(data) {
  if (typeof data !== 'object') return
  return new Observe(data)
}

3.3 建構式撰寫

 function Vue(options = {}) {
  //  模仿 Vue 把屬性掛載到 $options 且可以通過this._data訪問屬性
  this.$options = options
  const data = https://www.cnblogs.com/Jcloud/archive/2023/06/19/this._data = this.$options.data
  observe(data) // 給 data 增加資料劫持
}

上圖中我們模仿 Vue 對 options 和 data 增加了一些可訪問方式,給 data 增加了資料劫持,也在我們的實體中看到了效果,那這時又有了新的問題,Vue 中訪問資料都是 this.xxx 可直接通過實體訪問,這樣比我們圖中的訪問方式還要更方便一些,那我們也能否把屬性直接掛載到實體上呢,當然是可以的

3.4 資料代理

如果想要直接把屬性掛載到實體上,那我們需要保證通過實體直接訪問的屬性值是實時無誤的,且去修改該屬性值還能夠被劫持到,否則會影響后面的雙向資料系結,既然 data 中的資料我們已經通過 Observe(在3.2節)做了劫持,那我們在通過 this.xxx 直接修改屬性時只需要去修改 data 對應中的屬性就可以觸發 Observe 劫持

 function Vue(options = {}) {
  //  模仿 Vue 把屬性掛載到 $options
  this.$options = options
  const data = https://www.cnblogs.com/Jcloud/archive/2023/06/19/this._data = this.$options.data
  observe(data)
  // 將當前 this 傳入方法 將屬性掛載到 this 上
  proxyData.call(this, data)
}

// this 代理 this._data
function proxyData(data) {
  const vm = this
  for (let key in data) {
    Object.defineProperty(vm, key, {
      enumerable: true,
      get() {
        return vm._data[key]
      },
      set(newVal) {
// 直接修改 data 中對應屬性 觸發 data 中劫持 保持資料統一
        vm._data[key] = newVal 
      }
    })
  }
}

3.5 實作 compile

先在記憶體中創建一個檔案碎片來遞回所有 dom 節點,用正則匹配 {{}} 相符的節點,獲取到括號里的 key,最后在 data 中拿到對應 key 的屬性值,替換到節點上(因為主要是實作雙向系結,所以我們將 dom 的操作放到檔案碎片中操作來代替虛擬 dom)

function Compile(el, vm) {
  vm.$el = document.querySelector(el)
  // 建立檔案碎片 將 el 下的所有元素挪進檔案碎片 避免死回圈
  const fragment = document.createDocumentFragment()
  // 將 el 中的元素都移入碎片中
  while (child = vm.$el.firstChild) {
    fragment.appendChild(child)
  }
  // 匹配節點中的{{}} 將其替換為對應的值
  replace(fragment)
  function replace(fragment) {
    // 回圈每一層節點
    Array.from(fragment.childNodes).forEach(node => {
      const text = node.textContent
// 定義正則運算式
      const reg = /{{(.*)}}/
      // 此判斷為當節點是文本節點(因為變數都是文本)且被包含在{{}}中的文本節點時
// 文本節點 Node.TEXT_NODE: 3
      if (node.nodeType === Node.TEXT_NODE && reg.test(text)) {
// 以下三行為了獲取到 key 對應的value值 頁面初始化后正常將變數替換為值
        const arr = RegExp.$1.split('.') // [a, a]  [b]
        let val = vm
        arr.forEach(k => (val = val[k]))
        node.textContent = text.replace(reg, val)
      }
      if (node.childNodes) {
        replace(node)
      }
    })
  }
  // 將處理好的檔案碎片塞回dom中
  vm.$el.appendChild(fragment)
}

初始化 Vue 時呼叫 compile

 function Vue(options = {}) {
  //  模仿 Vue 把屬性掛載到 $options
  this.$options = options
  const data = https://www.cnblogs.com/Jcloud/archive/2023/06/19/this._data = this.$options.data
  observe(data)
  // 將當前 this 傳入方法 將屬性掛載到 this 上
  proxyData.call(this, data)
  new Compile(options.el, this)
}

3.6 Model -> ViewModel -> View

目前我們已經實作的功能:資料劫持、this代理、編譯模板 ,最終我們要達到修改資料、視圖自動更新的效果,還需要以下作業

1)第一步我們需要創建一個訂閱者,其 update 事件就是接收到我們更新后的資料值然后去更新 dom, 因為要更新 dom,所以此訂閱者是在 compile 中定義的,并且大家會發現我們在編譯程序中,是回圈每一層節點去判斷的,也就意味著我們頁面有多少個符合條件的文本節點,就會新建多少個 watcher,那這時就需要把文本節點的對應 key 和 value 傳入 watcher 中,用來判斷更新的哪個節點值

2)既然我們的 watcher 新增了引數(vue 實體、節點變數)所以我們需要對 watcher 方法做出更改

3)當 watcher 定義好后,還需要修改下其 update 方法,因為我們的 watcher 第三個引數也就是回呼函式中新增了引數,需要給其傳參

Watcher.prototype.update = function() {
  // this.exp 可取到 key 值 從 vm 中憑借 key 就可以取到屬性值
  let val = this.vm
  const arr = this.exp.split('.')
  arr.forEach(k => (val = val[k]))
  this.fn(val) // 傳入 newVal
}

4)訂閱者都準備好了,還需要添加訂閱者到 dep 陣列并且在資料改動后呼叫發布,這個程序需要在 observe 中實作

5)最終效果如圖所示

3.7 View -> ViewModel -> Model

上面我們實作了從資料到視圖的更新,那視圖從資料的更新呢,首先我們想到一個最常見的例子(v-model), 要想實作它,我們需要做以下兩步

1)把 value 值展示到系結 v-model 的 input 中

2)其次是每次我們更改值時都應該把值更新到界面上,所以我們還需要新建一個 watcher,在系結 v-model 的 input 上系結事件,當輸入文案時獲取到輸入的值,改變 data 只對應的屬性值

3)最終效果如圖所示

4 總結

4.1 實作思路

4.2 優缺點

優點:成功達到了資料和視圖的雙向驅動,像在操作表單時使用會更方便,省略了很多重復的 onChange 事件去處理資料的變化,也省略了給 dom 添加值的操作,代碼量會更少,更方便維護

缺點:修改資料時會使得我們無法追蹤資料的改變源頭,且在資料劫持那步需要去回圈,為一個物件的每一個屬性增加劫持,無法直接在一個物件上增加所有屬性的劫持(該缺點 Vue3 已規避,大家可自行學習)

4.3 小結

在作業中我們很多專案會用到框架 ,了解它的一些原理有助于我們更好的去使用,便于我們培養自己的‘造輪子’能力,遇到問題時能更好的解決,減少不必要的 bug,更好的去除錯代碼,一些很復雜的組件如果找不到開源的話,自己也能去實作不至于一頭霧水

4.4 參考資料

  • 可運行的該原始碼實體: https://coding.jd.com/zhangtingting155/mini-vue.git
  • Vue 雙向系結:https://github.com/answershuto/learnVue/blob/master/docs/從原始碼角度再看資料系結.MarkDown
  • 資料劫持:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty
  • 很有趣的設計模式:https://refactoringguru.cn/design-patterns

5 思考

至此,一個雙向資料系結功能就基本實作了,本文我們的實作是基于 Vue2 的雙向資料系結原理,目前 Vue3 已經趨于穩定,我們可以思考下,如果是基于 Vue3 的原理去做,那需要怎么去實作呢,

作者:京東物流 張婷婷

來源:京東云開發者社區

轉載請註明出處,本文鏈接:https://www.uj5u.com/qiye/555608.html

標籤:其他

上一篇:前端Vue圖片上傳組件支持單個檔案多個檔案上傳 自定義上傳數量 預覽洗掉圖片 圖片壓縮

下一篇:返回列表

標籤雲
其他(161326) Python(38242) JavaScript(25508) Java(18249) C(15237) 區塊鏈(8271) C#(7972) AI(7469) 爪哇(7425) MySQL(7258) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5875) 数组(5741) R(5409) Linux(5347) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4603) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2436) ASP.NET(2404) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1984) HtmlCss(1968) 功能(1967) Web開發(1951) C++(1942) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1881) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • IEEE1588PTP在數字化變電站時鐘同步方面的應用

    IEEE1588ptp在數字化變電站時鐘同步方面的應用 京準電子科技官微——ahjzsz 一、電力系統時間同步基本概況 隨著對IEC 61850標準研究的不斷深入,國內外學者提出基于IEC61850通信標準體系建設數字化變電站的發展思路。數字化變電站與常規變電站的顯著區別在于程序層傳統的電流/電壓互 ......

    uj5u.com 2020-09-10 03:51:52 more
  • HTTP request smuggling CL.TE

    CL.TE 簡介 前端通過Content-Length處理請求,通過反向代理或者負載均衡將請求轉發到后端,后端Transfer-Encoding優先級較高,以TE處理請求造成安全問題。 檢測 發送如下資料包 POST / HTTP/1.1 Host: ac391f7e1e9af821806e890 ......

    uj5u.com 2020-09-10 03:52:11 more
  • 網路滲透資料大全單——漏洞庫篇

    網路滲透資料大全單——漏洞庫篇漏洞庫 NVD ——美國國家漏洞庫 →http://nvd.nist.gov/。 CERT ——美國國家應急回應中心 →https://www.us-cert.gov/ OSVDB ——開源漏洞庫 →http://osvdb.org Bugtraq ——賽門鐵克 →ht ......

    uj5u.com 2020-09-10 03:52:15 more
  • 京準講述NTP時鐘服務器應用及原理

    京準講述NTP時鐘服務器應用及原理京準講述NTP時鐘服務器應用及原理 安徽京準電子科技官微——ahjzsz 北斗授時原理 授時是指接識訓通過某種方式獲得本地時間與北斗標準時間的鐘差,然后調整本地時鐘使時差控制在一定的精度范圍內。 衛星導航系統通常由三部分組成:導航授時衛星、地面檢測校正維護系統和用戶 ......

    uj5u.com 2020-09-10 03:52:25 more
  • 利用北斗衛星系統設計NTP網路時間服務器

    利用北斗衛星系統設計NTP網路時間服務器 利用北斗衛星系統設計NTP網路時間服務器 安徽京準電子科技官微——ahjzsz 概述 NTP網路時間服務器是一款支持NTP和SNTP網路時間同步協議,高精度、大容量、高品質的高科技時鐘產品。 NTP網路時間服務器設備采用冗余架構設計,高精度時鐘直接來源于北斗 ......

    uj5u.com 2020-09-10 03:52:35 more
  • 詳細解讀電力系統各種對時方式

    詳細解讀電力系統各種對時方式 詳細解讀電力系統各種對時方式 安徽京準電子科技官微——ahjzsz,更多資料請添加VX 衛星同步時鐘是我京準公司開發研制的應用衛星授時時技術的標準時間顯示和發送的裝置,該裝置以M國全球定位系統(GLOBAL POSITIONING SYSTEM,縮寫為GPS)或者我國北 ......

    uj5u.com 2020-09-10 03:52:45 more
  • 如何保證外包團隊接入企業內網安全

    不管企業規模的大小,只要企業想省錢,那么企業的某些服務就一定會采用外包的形式,然而看似美好又經濟的策略,其實也有不好的一面。下面我通過安全的角度來聊聊使用外包團的安全隱患問題。 先看看什么服務會使用外包的,最常見的就是話務/客服這種需要大量重復性、無技術性的服務,或者是一些銷售外包、特殊的職能外包等 ......

    uj5u.com 2020-09-10 03:52:57 more
  • PHP漏洞之【整型數字型SQL注入】

    0x01 什么是SQL注入 SQL是一種注入攻擊,通過前端帶入后端資料庫進行惡意的SQL陳述句查詢。 0x02 SQL整型注入原理 SQL注入一般發生在動態網站URL地址里,當然也會發生在其它地發,如登錄框等等也會存在注入,只要是和資料庫打交道的地方都有可能存在。 如這里http://192.168. ......

    uj5u.com 2020-09-10 03:55:40 more
  • [GXYCTF2019]禁止套娃

    git泄露獲取原始碼 使用GET傳參,引數為exp 經過三層過濾執行 第一層過濾偽協議,第二層過濾帶引數的函式,第三層過濾一些函式 preg_replace('/[a-z,_]+\((?R)?\)/', NULL, $_GET['exp'] (?R)參考當前正則運算式,相當于匹配函式里的引數 因此傳遞 ......

    uj5u.com 2020-09-10 03:56:07 more
  • 等保2.0實施流程

    流程 結論 ......

    uj5u.com 2020-09-10 03:56:16 more
最新发布
  • 手牽手帶你實作mini-vue

    Vue 的雙向資料系結實作原理是什么樣的,如果讓我們自己去實作一個這樣的雙向資料系結要怎么做呢,本文就與大家分享一下 Vue 的系結原理及其簡單實作 ......

    uj5u.com 2023-06-20 09:22:08 more
  • 前端Vue圖片上傳組件支持單個檔案多個檔案上傳 自定義上傳數量

    #### 前端Vue圖片上傳組件支持單個檔案多個檔案上傳 自定義上傳數量 預覽洗掉圖片 圖片壓縮, 下載完整代碼請訪問uni-app插件市場址:https://ext.dcloud.net.cn/plugin?id=13099 #### 效果圖如下: ![](https://p3-juejin.by ......

    uj5u.com 2023-06-20 09:22:04 more
  • StencilJs學習之組件裝飾器

    stenciljs 可以方便的構建互動式組件 支持以下裝飾器 - component - state - prop - watch - method - element - event - listen ## Component 裝飾器 `@Component` 是一個裝飾器,它將 TypeScri ......

    uj5u.com 2023-06-20 09:21:52 more
  • JavaScript的數學計算庫:decimal.js

    An arbitrary-precision Decimal type for JavaScript. ## 功能 - 整數和浮點數 - 簡單但功能齊全的 API - 復制 JavaScript 和物件的許多方法`Number.prototype` `Math` - 還處理十六進制、二進制和八進制值 ......

    uj5u.com 2023-06-20 09:21:46 more
  • JavaScript 顯示資料

    ## JavaScript 顯示資料 JavaScript 可以通過不同的方式來輸出資料: - 使用 **window.alert()** 彈出警告框。 - 使用 **document.write()** 方法將內容寫到 HTML 檔案中。 - 使用 **innerHTML** 寫入到 HTML 元 ......

    uj5u.com 2023-06-20 09:21:39 more
  • 管理軟體開發三分鐘入門

    利用藍點通用管理系統,可自定義各種管理功能,三分鐘入門,快速搭建各種資料管理/流程審批/資訊發布等功能,部署到云服務器,可隨時隨地用電腦或手機操作。支持自定義表單、流程、版式及圖表/報表,可接入微信和公眾號,輕松定制自己專屬的CRM系統、OA系統、HR系統,ERP等各種在線管理系統。 您的瀏覽器不支 ......

    uj5u.com 2023-06-19 08:38:56 more
  • 管理軟體開發三分鐘入門

    利用藍點通用管理系統,可自定義各種管理功能,三分鐘入門,快速搭建各種資料管理/流程審批/資訊發布等功能,部署到云服務器,可隨時隨地用電腦或手機操作。支持自定義表單、流程、版式及圖表/報表,可接入微信和公眾號,輕松定制自己專屬的CRM系統、OA系統、HR系統,ERP等各種在線管理系統。 您的瀏覽器不支 ......

    uj5u.com 2023-06-19 08:36:51 more
  • Vue 開發環境搭建

    ## 1 安裝環境 ### Node.js js的運行環境,相當于 java 的 jvm 官網:https://nodejs.org/en,下載最新穩定版 `18.16.0 LTS`,雙擊安裝即可 自動安裝了npm,終端驗證: ```bash C:\Users\Administrator>node ......

    uj5u.com 2023-06-19 08:22:27 more
  • React SSR - 寫個 Demo 一學就會

    # React SSR - 寫個 Demo 一學就會 今天寫個小 `Demo` 來從頭實作一下 `react` 的 `SSR`,幫助理解 `SSR` 是如何實作的,有什么細節。 ## 什么是 SSR `SSR` 即 `Server Side Rendering` 服務端渲染,是指將網頁內容在服務器端 ......

    uj5u.com 2023-06-19 08:17:12 more
  • 前端Vue非常簡單實用商品分類展示組件 側邊商品分類組件

    #### 前端vue非常簡單實用商品分類展示組件 側邊商品分類組件 , 下載完整代碼請訪問uni-app插件市場址:https://ext.dcloud.net.cn/plugin?id=13084 #### 效果圖如下: ![](https://p3-juejin.byteimg.com/tos- ......

    uj5u.com 2023-06-18 08:04:39 more