欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

vue mvvm數(shù)據(jù)響應(yīng)實(shí)現(xiàn)

 更新時(shí)間:2020年11月11日 09:05:30   作者:黑哪吒_blackNezha  
這篇文章主要介紹了vue mvvm數(shù)據(jù)響應(yīng)實(shí)現(xiàn)的方法,幫助大家更好的理解和使用vue,感興趣的朋友可以了解下

為什么實(shí)現(xiàn)數(shù)據(jù)響應(yīng)式

當(dāng)前vue、react等框架流行。無論是vue、還是react框架大家最初的設(shè)計(jì)思路都是類似的。都是以數(shù)據(jù)驅(qū)動(dòng)視圖,數(shù)據(jù)優(yōu)先。希望能夠通過框架減少開發(fā)人員直接操作節(jié)點(diǎn),讓開發(fā)人員能夠把更多的精力放在業(yè)務(wù)上而不是過多的放在操作節(jié)點(diǎn)上。另一方面,框架會(huì)通過虛擬dom及diff算法提高頁面性能。這其中需要數(shù)據(jù)優(yōu)先最根本的思路就是實(shí)現(xiàn)數(shù)據(jù)響應(yīng)式。so,本次來看下如何基于原生實(shí)現(xiàn)數(shù)據(jù)響應(yīng)式。

vue中的數(shù)據(jù)響應(yīng)

vue中會(huì)根據(jù)數(shù)據(jù)將數(shù)據(jù)通過大胡子語法及指令渲染到視圖上,這里我們以大胡子語法為例。如下:

<div id="app">
    {{message}}
</div>
let vm = new Vue({
  el:"#app",
  data:{
    message:"測試數(shù)據(jù)"
  }
})
setTimeout(()=>{
  vm.message = "修改的數(shù)據(jù)";
},1000)

如上代碼,很簡單 。vue做了兩件事情。一、把message數(shù)據(jù)初次渲染到視圖。二、當(dāng)message數(shù)據(jù)改變的時(shí)候視圖上渲染的message數(shù)據(jù)同時(shí)也會(huì)做出響應(yīng)。以最簡單的案例。帶著問題來看,通過原生js如何實(shí)現(xiàn)??這里為了簡化操作便于理解,這里就不去使用虛擬dom。直接操作dom結(jié)構(gòu)。

實(shí)現(xiàn)數(shù)據(jù)初次渲染

根據(jù)vue調(diào)用方式。定義Vue類來實(shí)現(xiàn)各種功能。將初次渲染過程定義成編譯compile函數(shù)渲染視圖。通過傳入的配置以及操作dom來實(shí)現(xiàn)渲染。大概思路是通過正則查找html 里 #app 作用域內(nèi)的表達(dá)式,然后查找數(shù)據(jù)做對應(yīng)的替換即可。具體實(shí)現(xiàn)如下:

class Vue {
  constructor(options) {
    this.opts = options;
    this.compile();
  }
  compile() {
    let ele = document.querySelector(this.opts.el);
    // 獲取所有子節(jié)點(diǎn)
    let childNodes = ele.childNodes;
    childNodes.forEach(node => {
      if (node.nodeType === 3) {
        // 找到所有的文本節(jié)點(diǎn)
        let nodeContent = node.textContent;
        // 匹配“{{}}”
        let reg = /\{\{\s*([^\{\}\s]+)\s*\}\}/g;
        if (reg.test(nodeContent)) {
          let $1 = RegExp.$1;
          // 查找數(shù)據(jù)替換 “{{}}”
          node.textContent = node.textContent.replace(reg, this.opts.data[$1]);
        }
      }
    })
  }
}

如上完成了初次渲染,將message數(shù)據(jù)渲染到了視圖上。但是會(huì)返現(xiàn)并沒對深層次的dom結(jié)構(gòu)做處理也就是如下情況:

 <div id="app">
    1{{ message }}2
    <div>
      hello , {{ message }}
    </div>
  </div>

渲染結(jié)果如上

發(fā)現(xiàn)結(jié)果并沒有達(dá)到預(yù)期。so,需要改下代碼,讓節(jié)點(diǎn)可以深層次查找就可以了。代碼如下:

  compile() {
    let ele = document.querySelector(this.opts.el);
    this.compileNodes(ele);
  }
  compileNodes(ele) {
    // 獲取所有子節(jié)點(diǎn)
    let childNodes = ele.childNodes;
    childNodes.forEach(node => {
      if (node.nodeType === 3) {
        // 找到所有的文本節(jié)點(diǎn)
        let nodeContent = node.textContent;
        // 匹配“{{}}”
        let reg = /\{\{\s*([^\{\}\s]+)\s*\}\}/g;
        if (reg.test(nodeContent)) {
          let $1 = RegExp.$1;
          // 查找數(shù)據(jù)替換 “{{}}”
          node.textContent = node.textContent.replace(reg, this.opts.data[$1]);
        }
      } else if (node.nodeType === 1) {
        if (node.childNodes.length > 0) {
          this.compileNodes(node);
        }
      }
    })
  }

上述代碼通過遞歸查找節(jié)點(diǎn) 實(shí)現(xiàn)深層次節(jié)點(diǎn)的渲染工作。如此,就實(shí)現(xiàn)了視圖的初次渲染。

數(shù)據(jù)劫持

回過頭來看下上面說的第二個(gè)問題:當(dāng)message數(shù)據(jù)改變的時(shí)候視圖上渲染的message數(shù)據(jù)同時(shí)也會(huì)做出響應(yīng)。如何實(shí)現(xiàn)數(shù)據(jù)響應(yīng)式?簡而言之就是數(shù)據(jù)變動(dòng)影響視圖變動(dòng)?再將問題拆分下 1. 如何知道數(shù)據(jù)變動(dòng)了? 2.如何根據(jù)數(shù)據(jù)變動(dòng)來更改視圖?

  • 如何知道數(shù)據(jù)變動(dòng)了? 這里就需要用到數(shù)據(jù)攔截了,或者叫數(shù)據(jù)觀察。把會(huì)變動(dòng)的data數(shù)據(jù)觀察起來。當(dāng)他變動(dòng)的時(shí)候我們可以做后續(xù)的渲染事情。如何攔截?cái)?shù)據(jù)呢 ?vue2里采取的是definePrototype。
let obj = {
  myname:"張三"
}
Object.defineProperty(obj,'myname',{
  configurable:true,
  enumerable:true,
  get(){
    console.log("get.")
    return "張三";
  },
  set(newValue){
    console.log("set")
    console.log(newValue);
  }
})
console.log(obj);

上述代碼會(huì)發(fā)現(xiàn),通過defineProperty劫持的對象屬性下都會(huì)有g(shù)et及set方法。那么當(dāng)我們獲取或者設(shè)置數(shù)據(jù)的時(shí)候就能出發(fā)對應(yīng)的get及set 。這樣就能攔截?cái)?shù)據(jù)做后續(xù)操作。

還有沒有其他方式達(dá)到數(shù)據(jù)劫持的效果呢?ES6中出現(xiàn)了Proxy 代理對象同樣也可以達(dá)到類似劫持?jǐn)?shù)據(jù)的功能。如下代碼:

let obj = {
  myname:"張三"
}
let newObj = new Proxy(obj,{
  get(target,key){
    console.log("get...")
    return "張三"
  },
  set(target,name,newValue){
    console.log("set...");
    return Reflect.set(target,name,newValue);
  }
})

兩種方式都可以實(shí)現(xiàn)數(shù)據(jù)劫持。proxy功能更加強(qiáng)大,很多方法是defineProperty所不具備的。且proxy直接攔截的是對象而defineProperty攔截的是對象屬性。so,可以利用上述方式將data數(shù)據(jù)做劫持,代碼如下:

observe(data){
    let keys = Object.keys(data);
    keys.forEach(key=>{
      let value = data[key];
      Object.defineProperty(data,key,{
        configurable:true,
        enumerable:true,
        get(){
          return value;
        },
        set(newValue){
          value = newValue;
        }
      });
    })
 }

觀察者模式實(shí)現(xiàn)數(shù)據(jù)響應(yīng)

有了劫持?jǐn)?shù)據(jù)方式后,接下來需要實(shí)現(xiàn)的就是當(dāng)修改數(shù)據(jù)的時(shí)候?qū)⑿聰?shù)據(jù)渲染到視圖。如何辦到呢?會(huì)發(fā)現(xiàn),需要在data設(shè)置的時(shí)候觸發(fā)視圖的compile編譯。二者之間互相影響,此時(shí)可以想到利用觀察者模式,通過觀察者模式讓二者產(chǎn)生關(guān)聯(lián),如下:

圖略小,代碼也貼上吧。

class Vue extends EventTarget {
  constructor(options) {
    super();
    this.opts = options;
    this.observe(this.opts.data);
    this.compile();
  }
  observe(data){
    let keys = Object.keys(data);
    let _this = this;
    keys.forEach(key=>{
      let value = data[key];
      Object.defineProperty(data,key,{
        configurable:true,
        enumerable:true,
        get(){
          return value;
        },
        set(newValue){
          _this.dispatchEvent(new CustomEvent(key,{
            detail:newValue
          }));
          value = newValue;
        }
      });
    })
  }
  compile() {
    let ele = document.querySelector(this.opts.el);
    this.compileNodes(ele);
  }
  compileNodes(ele) {
    // 獲取所有子節(jié)點(diǎn)
    let childNodes = ele.childNodes;
    childNodes.forEach(node => {
      if (node.nodeType === 3) {
        // 找到所有的文本節(jié)點(diǎn)
        let nodeContent = node.textContent;
        // 匹配“{{}}”
        let reg = /\{\{\s*([^\{\}\s]+)\s*\}\}/g;
        if (reg.test(nodeContent)) {
          let $1 = RegExp.$1;
          // 查找數(shù)據(jù)替換 “{{}}”
          node.textContent = node.textContent.replace(reg, this.opts.data[$1]);
          this.addEventListener($1,e=>{
            let oldValue = this.opts.data[$1];
            let newValue = e.detail;
            let reg = new RegExp(oldValue);
            node.textContent = node.textContent.replace(reg,newValue);
          })
        }
      } else if (node.nodeType === 1) {
        if (node.childNodes.length > 0) {
          this.compileNodes(node);
        }
      }
    })
  }
}

如上,成功的通過觀察者模式實(shí)現(xiàn)了數(shù)據(jù)的響應(yīng)。但是會(huì)發(fā)現(xiàn)data與compile之間需要通過鍵名來進(jìn)行關(guān)聯(lián)。如果data數(shù)據(jù)結(jié)構(gòu)嵌套關(guān)系復(fù)雜后面會(huì)比較難處理。有沒有一種方式讓二者松解耦呢?這時(shí)候可以用發(fā)布訂閱模式來進(jìn)行改造。

發(fā)布訂閱模式改造響應(yīng)式

還是略小,也還是貼上代碼:

class Vue {
  constructor(options) {
    this.opts = options;
    this.observe(this.opts.data);
    this.compile();
  }
  observe(data){
    let keys = Object.keys(data);
    let _this = this;
    keys.forEach(key=>{
      let value = data[key];
      let dep = new Dep();
      Object.defineProperty(data,key,{
        configurable:true,
        enumerable:true,
        get(){
          if(Dep.target){
            dep.addSub(Dep.target); 
          }
          return value;
        },
        set(newValue){
          dep.notify(newValue);
          value = newValue;
        }
      });
    })
  }
  compile() {
    let ele = document.querySelector(this.opts.el);
    this.compileNodes(ele);
  }
  compileNodes(ele) {
    // 獲取所有子節(jié)點(diǎn)
    let childNodes = ele.childNodes;
    childNodes.forEach(node => {
      if (node.nodeType === 3) {
        // 找到所有的文本節(jié)點(diǎn)
        let nodeContent = node.textContent;
        // 匹配“{{}}”
        let reg = /\{\{\s*([^\{\}\s]+)\s*\}\}/g;
        if (reg.test(nodeContent)) {
          let $1 = RegExp.$1;
          // 查找數(shù)據(jù)替換 “{{}}”
          node.textContent = node.textContent.replace(reg, this.opts.data[$1]);
          new Watcher(this.opts.data,$1,(newValue)=>{
            let oldValue = this.opts.data[$1];
            let reg = new RegExp(oldValue);
            node.textContent = node.textContent.replace(reg,newValue);
          })
        }
      } else if (node.nodeType === 1) {
        if (node.childNodes.length > 0) {
          this.compileNodes(node);
        }
      }
    })
  }
}

class Dep{
  constructor(){
    this.subs = [];
  }
  addSub(sub){
    this.subs.push(sub);
  }
  notify(newValue){
    this.subs.forEach(sub=>{
      sub.update(newValue);
    })
  }
}

class Watcher{
  constructor(data,key,cb){
    Dep.target = this;
    data[key];
    this.cb = cb;
    Dep.target = null;
  }
  update(newValue){
    this.cb(newValue);
  }
}

如上代碼思路是 針對每個(gè)數(shù)據(jù)會(huì)生成一個(gè)dep(依賴收集器)在數(shù)據(jù)get的時(shí)候收集watcher,將watcher 添加到dep里保存。數(shù)據(jù)一旦有改變觸發(fā)notify發(fā)布消息從而影響compile編譯更新視圖。這個(gè)流程也可以參看下圖:

如上就完成了視圖響應(yīng)。通過上述代碼,我們可以看出實(shí)現(xiàn)數(shù)據(jù)響應(yīng)兩個(gè)核心點(diǎn)1.數(shù)據(jù)劫持。2.觀察者和發(fā)布訂閱。在這我們可以思考一個(gè)問題,2個(gè)設(shè)計(jì)模式都是可以實(shí)現(xiàn)的但是有什么區(qū)別呢?

觀察者與發(fā)布訂閱

這里需要從概念來看

  • 觀察者模式:定義一個(gè)對象與其他對象之間的一種依賴關(guān)系,當(dāng)對象發(fā)生某種變化的時(shí)候,依賴它的其它對象都會(huì)得到更新,一對多的關(guān)系。
  • 發(fā)布訂閱模式:是一種消息范式,消息的發(fā)送者(稱為發(fā)布者)不會(huì)將消息直接發(fā)送給特定的接收者(稱為訂閱者)。而是將發(fā)布的消息分為不同的類別,無需了解哪些訂閱者(如果有的話)可能存在。同樣的,訂閱者可以表達(dá)對一個(gè)或多個(gè)類別的興趣,只接收感興趣的消息,無需了解哪些發(fā)布者(如果有的話)存在。

兩者之間關(guān)系,發(fā)布訂閱是三者之間關(guān)系。發(fā)布訂閱會(huì)多了一個(gè)關(guān)系器來組織主題和觀察者之間的關(guān)系。這樣做的好處就是松解耦??瓷厦骓憫?yīng)式例子可以看出觀察者需要通過事件名稱來進(jìn)行關(guān)聯(lián)。發(fā)布訂閱定義dep管理器之后data和compile徹底解耦,讓二者松散解耦。在處理多層數(shù)據(jù)結(jié)構(gòu)上發(fā)布訂閱會(huì)更清晰。松解耦能夠應(yīng)對更多變化,把模塊之間依賴降到最低。發(fā)布訂閱廣義上是觀察者模式。

好了 暫時(shí)先over 。 如果覺得有收獲的話可以點(diǎn)個(gè)贊,贈(zèng)人玫瑰,手有余香?。。。?/p>

以上就是vue mvvm數(shù)據(jù)響應(yīng)實(shí)現(xiàn)的詳細(xì)內(nèi)容,更多關(guān)于vue mvvm數(shù)據(jù)響應(yīng)的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Vue常用的修飾符及應(yīng)用場景解讀

    Vue常用的修飾符及應(yīng)用場景解讀

    這篇文章主要介紹了Vue常用的修飾符及應(yīng)用場景解讀,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • Vue3.3?+?TS4構(gòu)建實(shí)現(xiàn)ElementPlus功能的組件庫示例

    Vue3.3?+?TS4構(gòu)建實(shí)現(xiàn)ElementPlus功能的組件庫示例

    Vue.js?是目前最盛行的前端框架之一,而?TypeScript?則是一種靜態(tài)類型言語,它能夠讓開發(fā)人員在編寫代碼時(shí)愈加平安和高效,本文將引見如何運(yùn)用?Vue.js?3.3?和?TypeScript?4?構(gòu)建一個(gè)自主打造媲美?ElementPlus?的組件庫
    2023-10-10
  • Vue彈窗組件的實(shí)現(xiàn)方法

    Vue彈窗組件的實(shí)現(xiàn)方法

    這篇文章主要為大家詳細(xì)介紹了Vue彈窗組件的實(shí)現(xiàn)方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-09-09
  • vue-print-nb解決vue打印問題,并且隱藏頁眉頁腳方式

    vue-print-nb解決vue打印問題,并且隱藏頁眉頁腳方式

    這篇文章主要介紹了vue-print-nb解決vue打印問題,并且隱藏頁眉頁腳方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • vue 地區(qū)選擇器v-distpicker的常用功能

    vue 地區(qū)選擇器v-distpicker的常用功能

    這篇文章主要介紹了vue 地區(qū)選擇器v-distpicker的常用功能,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2019-07-07
  • vue實(shí)現(xiàn)登錄類型切換

    vue實(shí)現(xiàn)登錄類型切換

    這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)登錄類型切換,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-04-04
  • Vue數(shù)據(jù)變了但頁面沒有變的幾種情況及解決方法

    Vue數(shù)據(jù)變了但頁面沒有變的幾種情況及解決方法

    如果,你發(fā)現(xiàn)自己需要在Vue中做一次強(qiáng)制更新,99.99%的情況,是你在某個(gè)地方做錯(cuò)了事,本文給大家就介紹了Vue數(shù)據(jù)變了,但頁面沒有變的幾種情況及解決方法,并通過代碼示例介紹的非常詳細(xì),需要的朋友可以參考下
    2024-08-08
  • webpack+vue-cli項(xiàng)目中引入外部非模塊格式j(luò)s的方法

    webpack+vue-cli項(xiàng)目中引入外部非模塊格式j(luò)s的方法

    今天小編就為大家分享一篇webpack+vue-cli項(xiàng)目中引入外部非模塊格式j(luò)s的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-09-09
  • vue中v-for 循環(huán)對象中的屬性

    vue中v-for 循環(huán)對象中的屬性

    這篇文章主要介紹了 vue中v-for 循環(huán)對象中的屬性,文章圍繞v-for 循環(huán)對象的相關(guān)資料展開詳細(xì)內(nèi)容,需要的朋友可以參考一下,希望對大家有所幫助
    2021-11-11
  • Vue攔截器原理以及詳細(xì)使用方法

    Vue攔截器原理以及詳細(xì)使用方法

    這篇文章主要給大家介紹了關(guān)于Vue攔截器原理以及詳細(xì)使用的相關(guān)資料,Vue攔截器通常用于在發(fā)送請求或響應(yīng)數(shù)據(jù)時(shí)對其進(jìn)行一些處理或修改,文中通過代碼示例介紹的非常詳細(xì),需要的朋友可以參考下
    2023-08-08

最新評(píng)論