真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網站制作重慶分公司

vue雙向綁定原理實例分析

這篇文章主要介紹了vue雙向綁定原理實例分析的相關知識,內容詳細易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇vue雙向綁定原理實例分析文章都會有所收獲,下面我們一起來看看吧。

專注于為中小企業(yè)提供網站制作、網站設計服務,電腦端+手機端+微信端的三站合一,更高效的管理,為中小企業(yè)東興免費做網站提供優(yōu)質的服務。我們立足成都,凝聚了一批互聯(lián)網行業(yè)人才,有力地推動了成百上千家企業(yè)的穩(wěn)健成長,幫助中小企業(yè)通過網站建設實現(xiàn)規(guī)模擴充和轉變。

vue雙向綁定原理實例分析

自定義vue類

  • vue最少需要兩個參數(shù):模板和data。

  • 創(chuàng)建Compiler對象,將數(shù)據(jù)渲染到模板后,掛載到指定跟節(jié)點中。

class MyVue {
  // 1,接收兩個參數(shù):模板(根節(jié)點),和數(shù)據(jù)對象
  constructor(options) {
    // 保存模板,和數(shù)據(jù)對象
    if (this.isElement(options.el)) {
      this.$el = options.el;
    } else {
      this.$el = document.querySelector(options.el);
    }
    this.$data = options.data;
    // 2.根據(jù)模板和數(shù)據(jù)對象,渲染到根節(jié)點
    if (this.$el) {
      // 監(jiān)聽data所有屬性的get/set
      new Observer(this.$data);
      new Compiler(this)
    }
  }
  // 判斷是否是一個dom元素
  isElement(node) {
    return node.nodeType === 1;
  }
}

實現(xiàn)數(shù)據(jù)首次渲染到頁面

Compiler

1,node2fragment函數(shù)將模板元素提取到內存中,方便將數(shù)據(jù)渲染到模板后,再一次性掛載到頁面中

2,模板提取到內存后,使用buildTemplate函數(shù)遍歷該模板元素

  • 元素節(jié)點

    • 使用buildElement函數(shù)檢查元素上以v-開頭的屬性

  • 文本節(jié)點

    • 用buildText函數(shù)檢查文本中有無{{}}內容

3,創(chuàng)建CompilerUtil類,用于處理vue指令和{{}},完成數(shù)據(jù)的渲染

4,到此就完成了首次數(shù)據(jù)渲染,接下來需要實現(xiàn)數(shù)據(jù)改變時,自動更新視圖。

class Compiler {
  constructor(vm) {
    this.vm = vm;
    // 1.將網頁上的元素放到內存中
    let fragment = this.node2fragment(this.vm.$el);
    // 2.利用指定的數(shù)據(jù)編譯內存中的元素
    this.buildTemplate(fragment);
    // 3.將編譯好的內容重新渲染會網頁上
    this.vm.$el.appendChild(fragment);
  }
  node2fragment(app) {
    // 1.創(chuàng)建一個空的文檔碎片對象
    let fragment = document.createDocumentFragment();
    // 2.編譯循環(huán)取到每一個元素
    let node = app.firstChild;
    while (node) {
      // 注意點: 只要將元素添加到了文檔碎片對象中, 那么這個元素就會自動從網頁上消失
      fragment.appendChild(node);
      node = app.firstChild;
    }
    // 3.返回存儲了所有元素的文檔碎片對象
    return fragment;
  }
  buildTemplate(fragment) {
    let nodeList = [...fragment.childNodes];
    nodeList.forEach(node => {
      // 需要判斷當前遍歷到的節(jié)點是一個元素還是一個文本
      if (this.vm.isElement(node)) {
        // 元素節(jié)點
        this.buildElement(node);
        // 處理子元素
        this.buildTemplate(node);
      } else {
        // 文本節(jié)點
        this.buildText(node);
      }
    })
  }
  buildElement(node) {
    let attrs = [...node.attributes];
    attrs.forEach(attr => {
      // v-model="name" => {name:v-model  value:name}
      let { name, value } = attr;
      // v-model / v-html / v-text / v-xxx
      if (name.startsWith('v-')) {
        // v-model -> [v, model]
        let [_, directive] = name.split('-');
        CompilerUtil[directive](node, value, this.vm);
      }
    })
  }
  buildText(node) {
    let content = node.textContent;
    let reg = /\{\{.+?\}\}/gi;
    if (reg.test(content)) {
      CompilerUtil['content'](node, content, this.vm);
    }
  }
}
let CompilerUtil = {
  getValue(vm, value) {
    // 解析this.data.aaa.bbb.ccc這種屬性
    return value.split('.').reduce((data, currentKey) => {
      return data[currentKey.trim()];
    }, vm.$data);
  },
  getContent(vm, value) {
    // 解析{{}}中的變量
    let reg = /\{\{(.+?)\}\}/gi;
    let val = value.replace(reg, (...args) => {
      return this.getValue(vm, args[1]);
    });
    return val;
  },
  // 解析v-model指令
  model: function (node, value, vm) {
    // 在觸發(fā)getter之前,為dom創(chuàng)建Wather,并為Watcher.target賦值
    new Watcher(vm, value, (newValue, oldValue) => {
      node.value = newValue;
    });
    let val = this.getValue(vm, value);
    node.value = val;
  },
  // 解析v-html指令
  html: function (node, value, vm) {
    // 在觸發(fā)getter之前,為dom創(chuàng)建Wather,并為Watcher.target賦值
    new Watcher(vm, value, (newValue, oldValue) => {
      node.innerHTML = newValue;
    });
    let val = this.getValue(vm, value);
    node.innerHTML = val;
  },
  // 解析v-text指令
  text: function (node, value, vm) {
    // 在觸發(fā)getter之前,為dom創(chuàng)建Wather,并為Watcher.target賦值
    new Watcher(vm, value, (newValue, oldValue) => {
      node.innerText = newValue;
    });
    let val = this.getValue(vm, value);
    node.innerText = val;
  },
  // 解析{{}}中的變量
  content: function (node, value, vm) {
    let reg = /\{\{(.+?)\}\}/gi;
    let val = value.replace(reg, (...args) => {
      // 在觸發(fā)getter之前,為dom創(chuàng)建Wather,并為Watcher.target賦值
      new Watcher(vm, args[1], (newValue, oldValue) => {
        node.textContent = this.getContent(vm, value);
      });
      return this.getValue(vm, args[1]);
    });
    node.textContent = val;
  }
}

實現(xiàn)數(shù)據(jù)驅動視圖

Observer

1,使用defineRecative函數(shù)對data做Object.defineProperty處理,使得data中的每個數(shù)據(jù)都可以進行get/set監(jiān)聽

2,接下來將考慮如何在監(jiān)聽到data值改變后,更新視圖內容呢?使用觀察者設計模式,創(chuàng)建Dep和Wather類。

class Observer {
  constructor(data) {
    this.observer(data);
  }
  observer(obj) {
    if (obj && typeof obj === 'object') {
      // 遍歷取出傳入對象的所有屬性, 給遍歷到的屬性都增加get/set方法
      for (let key in obj) {
        this.defineRecative(obj, key, obj[key])
      }
    }
  }
  // obj: 需要操作的對象
  // attr: 需要新增get/set方法的屬性
  // value: 需要新增get/set方法屬性的取值
  defineRecative(obj, attr, value) {
    // 如果屬性的取值又是一個對象, 那么也需要給這個對象的所有屬性添加get/set方法
    this.observer(value);
    // 第三步: 將當前屬性的所有觀察者對象都放到當前屬性的發(fā)布訂閱對象中管理起來
    let dep = new Dep(); // 創(chuàng)建了屬于當前屬性的發(fā)布訂閱對象
    Object.defineProperty(obj, attr, {
      get() {
        // 在這里收集依賴
        Dep.target && dep.addSub(Dep.target);
        return value;
      },
      set: (newValue) => {
        if (value !== newValue) {
          // 如果給屬性賦值的新值又是一個對象, 那么也需要給這個對象的所有屬性添加get/set方法
          this.observer(newValue);
          value = newValue;
          dep.notify();
          console.log('監(jiān)聽到數(shù)據(jù)的變化');
        }
      }
    })
  }
}

使用觀察者設計模式,創(chuàng)建Dep和Wather類

1,使用觀察者設計模式的目的是:

  • 解析模板,收集data中某個數(shù)據(jù)在模板中被使用的dom節(jié)點集合,當該數(shù)據(jù)改變時,更新該dom節(jié)點集合就實現(xiàn)了數(shù)據(jù)更新。

  • Dep:用于收集某個data屬性依賴的dom節(jié)點集合,并提供更新方法

  • Watcher:每個dom節(jié)點的包裹對象

    • attr:該dom使用的data屬性

    • cb:修改該dom值的回調函數(shù),在創(chuàng)建的時候會接收

2,到這里感覺思路是沒問題了,已經是勝券在握了。那Dep和Watcher該怎么使用呢?

  • 為每個屬性添加一個dep,用來收集依賴的dom

  • 因為頁面首次渲染的時候會讀取data數(shù)據(jù),這時候會觸發(fā)該data的getter,所以在此收集dom

  • 具體如何收集呢,在CompilerUtil類解析v-model,{{}}等命令時,會觸發(fā)getter,我們在觸發(fā)之前創(chuàng)建Wather,為Watcher添加一個靜態(tài)屬性,指向該dom,然后在getter函數(shù)里面獲取該靜態(tài)變量,并添加到依賴中,就完成了一次收集。因為每次觸發(fā)getter之前都對該靜態(tài)變量賦值,所以不存在收集錯依賴的情況。

class Dep {
  constructor() {
    // 這個數(shù)組就是專門用于管理某個屬性所有的觀察者對象的
    this.subs = [];
  }
  // 訂閱觀察的方法
  addSub(watcher) {
    this.subs.push(watcher);
  }
  // 發(fā)布訂閱的方法
  notify() {
    this.subs.forEach(watcher => watcher.update());
  }
}
class Watcher {
  constructor(vm, attr, cb) {
    this.vm = vm;
    this.attr = attr;
    this.cb = cb;
    // 在創(chuàng)建觀察者對象的時候就去獲取當前的舊值
    this.oldValue = this.getOldValue();
  }
  getOldValue() {
    Dep.target = this;
    let oldValue = CompilerUtil.getValue(this.vm, this.attr);
    Dep.target = null;
    return oldValue;
  }
  // 定義一個更新的方法, 用于判斷新值和舊值是否相同
  update() {
    let newValue = CompilerUtil.getValue(this.vm, this.attr);
    if (this.oldValue !== newValue) {
      this.cb(newValue, this.oldValue);
    }
  }
}

3,到這里就實現(xiàn)了數(shù)據(jù)綁定時,視圖自動更新,本來想代碼一步步實現(xiàn)的,但是發(fā)現(xiàn)不好處理,就把完整的class貼出來了。

實現(xiàn)視圖驅動數(shù)據(jù)

其實就是監(jiān)聽輸入框的input、change事件。修改CompilerUtil的model方法。具體代碼如下

model: function (node, value, vm) {
    new Watcher(vm, value, (newValue, oldValue)=>{
        node.value = newValue;
    });
    let val = this.getValue(vm, value);
    node.value = val;
	// 看這里
    node.addEventListener('input', (e)=>{
        let newValue = e.target.value;
        this.setValue(vm, value, newValue);
    })
},

總結

vue雙向綁定原理

vue接收一個模板和data參數(shù)。

1,首先將data中的數(shù)據(jù)進行遞歸遍歷,對每個屬性執(zhí)行Object.defineProperty,定義get和set函數(shù)。并為每個屬性添加一個dep數(shù)組。當get執(zhí)行時,會為調用的dom節(jié)點創(chuàng)建一個watcher存放在該數(shù)組中。當set執(zhí)行時,重新賦值,并調用dep數(shù)組的notify方法,通知所有使用了該屬性watcher,并更新對應dom的內容。

2,將模板加載到內存中,遞歸模板中的元素,檢測到元素有v-開頭的命令或者雙大括號的指令,就會從data中取對應的值去修改模板內容,這個時候就將該dom元素添加到了該屬性的dep數(shù)組中。這就實現(xiàn)了數(shù)據(jù)驅動視圖。在處理v-model指令的時候,為該dom添加input事件(或change),輸入時就去修改對應的屬性的值,實現(xiàn)了頁面驅動數(shù)據(jù)。

3,將模板與數(shù)據(jù)進行綁定后,將模板添加到真實dom樹中。

如何將watcher放在dep數(shù)組中?

在解析模板的時候,會根據(jù)v-指令獲取對應data屬性值,這個時候就會調用屬性的get方法,我們先創(chuàng)建Watcher實例,并在其內部獲取該屬性值,作為舊值存放在watcher內部,我們在獲取該值之前,在Watcher原型對象上添加屬性Watcher.target = this;然后取值,將講Watcher.target = null;這樣get在被調用的時候就可以根據(jù)Watcher.target獲取到watcher實例對象。

methods的原理

創(chuàng)建vue實例的時候,接收methods參數(shù)

在解析模板的時候遇到v-on的指令。會對該dom元素添加對應事件的監(jiān)聽,并使用call方法將vue綁定為該方法的this:vm.$methods[value].call(vm, e);

computed的原理

創(chuàng)建vue實例的時候,接收computed參數(shù)

初始化vue實例的時候,為computed的key進行Object.defineProperty處理,并添加get屬性。

關于“vue雙向綁定原理實例分析”這篇文章的內容就介紹到這里,感謝各位的閱讀!相信大家對“vue雙向綁定原理實例分析”知識都有一定的了解,大家如果還想學習更多知識,歡迎關注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


當前文章:vue雙向綁定原理實例分析
新聞來源:http://weahome.cn/article/jsphpo.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部