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

淺析Proxy可以優(yōu)化vue的數(shù)據(jù)監(jiān)聽機(jī)制問題及實現(xiàn)思路

 更新時間:2018年11月29日 08:14:02   作者:南柯一夢同志  
這篇文章主要介紹了淺析Proxy可以優(yōu)化vue的數(shù)據(jù)監(jiān)聽機(jī)制問題及實現(xiàn)思路,需要的朋友可以參考下

我們首先來看vue2.x中的實現(xiàn),為簡單起見,我們這里不考慮多級嵌套,也不考慮數(shù)組

vue2.x中的實現(xiàn)

其本質(zhì)是new Watcher(data, key, callback)的方式,而在調(diào)用之前是先將data中的所有屬性轉(zhuǎn)化成可監(jiān)聽的對象, 其主要就是利用Object.defineProperty,。

class Watcher{
  constructor(data, key, cb){
  }
}
//轉(zhuǎn)換成可監(jiān)聽對象
function observe(data){
  new Observer(data)
}
//修改數(shù)據(jù)的getter和setter
function defineReactive(obj, key){
  let value = obj[key];
  Object.defineProperty(obj, key, {
    enumerable: true,
    configurable: true,
    get(){
      return value;
    },
    set(newVal){
      value = newVal
    }
  })
}

Observer的實現(xiàn)很簡單

class Observer {
  constructor(data){
    this.walk(data);
  }

  walk(data){
    for(var key in data) {
      // 這里不考慮嵌套的問題,否則的話需要遞歸調(diào)用walk
      defineReactive(data, key)
    }
  }
}

現(xiàn)在怎么將watcher和getter/setter聯(lián)系起來,vue的方法是添加一個依賴類:Dep

class Watcher{
  constructor(data, key, cb){
    this.cb = cb;
    Dep.target = this; //每次新建watcher的時候講給target賦值,對target的管理這里簡化了vue的實現(xiàn)
    data[key];//調(diào)用getter,執(zhí)行addSub, 將target傳入對應(yīng)的dep; vue的實現(xiàn)本質(zhì)就是如此
  }
}
class Dep {
  constructor(){
    this.subs = [];
  }
  addSub(sub){
    this.subs.push(sub);
  }
  notify(){
    this.subs.forEach(sub => sub.cb())
  }
}
function defineReactive(obj, key){
  let value = obj[key];
  let dep = new Dep(); //每一個屬性都有一個對應(yīng)的dep,作為閉包保存
  Object.defineProperty(obj, key, {
    enumerable: true,
    configurable: true,
    get(){
      dep.addSub(Dep.target)
      Dep.target = null;
      return value;
    },
    set(newVal){
      value = newVal
      dep.notify();
    }
  })
}

以上就是vue的思路,vue3之所以要從新實現(xiàn),主要有這幾個原因:

  1. Object.defineProperty的性能開銷。
  2. defineReactive一開始就要對要監(jiān)聽的對象所有屬性都執(zhí)行一遍,因為傳統(tǒng)方法要將一個對象轉(zhuǎn)換成可監(jiān)聽對象,只能如此。
  3. 添加刪除屬性的問題。
  4. 還有一點就是這個模塊被耦合到了vue里面,新版本可以單獨作為一個庫來使用。

然后我們來看看同樣的功能采用Proxy會怎樣實現(xiàn)。

Proxy的實現(xiàn)

將一個對象轉(zhuǎn)換成Proxy的方式很簡單,只需要作為參數(shù)傳給proxy即可;

class Watcher {
  constructor(proxy, key, cb) {
    this.cb = cb;
    Dep.target = this;
    this.value = proxy[key];
  }
}
class Dep {
  constructor(){
    this.subs = []
  }
  addSub(sub){
    this.subs.push(sub);
  }
  notify(newVal){
    this.subs.forEach(sub => {
      sub.cb(newVal, sub.value);
      sub.value = newVal;
    })
  }
}
const observe = (obj) => {
  const deps = {};
  return new Proxy(obj, {
    get: function (target, key, receiver) {
      const dep = (deps[key] = deps[key] || new Dep);
      Dep.target && dep.addSub(Dep.target)
      Dep.target = null;
      return Reflect.get(target, key, receiver);
    },
    set: function (target, key, value, receiver) {
      const dep = (deps[key] = deps[key] || new Dep);
      Promise.resolve().then(() => {
        dep.notify(value);
      })
      return Reflect.set(target, key, value, receiver);
    }
  });
}
var state = observe({x:0})
new Watcher(state, 'x', function(n, o){
  console.log(n, o)
});
new Watcher(state, 'y', function(n, o){
  console.log(n, o)
});
state.x = 3;
state.y = 3;

也許一開始我們只關(guān)心x和y,那么就不會對其他的屬性做相應(yīng)的處理,除非添加watcher,其他時間target都是null

總結(jié)

以上所述是小編給大家介紹的Proxy可以優(yōu)化vue的數(shù)據(jù)監(jiān)聽機(jī)制問題,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!

相關(guān)文章

  • Vue圖片裁剪組件實例代碼

    Vue圖片裁剪組件實例代碼

    這篇文章主要給大家介紹了關(guān)于Vue圖片裁剪組件的相關(guān)資料,本文介紹的組件是基于vue-cropper二次封裝,vue-cropper大家應(yīng)該都很熟悉了吧,需要的朋友可以參考下
    2021-07-07
  • Vue實現(xiàn)active點擊切換方法

    Vue實現(xiàn)active點擊切換方法

    下面小編就為大家分享一篇Vue實現(xiàn)active點擊切換方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-03-03
  • vue實現(xiàn)表單驗證功能

    vue實現(xiàn)表單驗證功能

    這篇文章主要為大家詳細(xì)介紹了vue實現(xiàn)表單驗證功能,基于NUXT的validate方法實現(xiàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • Vue+node實現(xiàn)音頻錄制播放功能

    Vue+node實現(xiàn)音頻錄制播放功能

    這篇文章主要介紹了Vue+node實現(xiàn)音頻錄制播放,功能,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-03-03
  • vue實現(xiàn)記事本功能

    vue實現(xiàn)記事本功能

    這篇文章主要為大家詳細(xì)介紹了vue實現(xiàn)記事本功能,記事本可以進(jìn)行添加刪除事件,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-06-06
  • 詳解Vue 動態(tài)添加模板的幾種方法

    詳解Vue 動態(tài)添加模板的幾種方法

    本篇文章主要介紹了詳解Vue 動態(tài)添加模板的幾種方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-04-04
  • Vue.js上下滾動加載組件的實例代碼

    Vue.js上下滾動加載組件的實例代碼

    本篇文章主要介紹了Vue.js上下滾動加載組件的實例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-07-07
  • 15 分鐘掌握vue-next響應(yīng)式原理

    15 分鐘掌握vue-next響應(yīng)式原理

    這篇文章主要介紹了15 分鐘掌握vue-next響應(yīng)式原理,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-10-10
  • vue 彈出遮罩層樣式實例

    vue 彈出遮罩層樣式實例

    這篇文章主要介紹了vue 彈出遮罩層樣式實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-07-07
  • vue前端項目打包成Docker鏡像并運(yùn)行的實現(xiàn)

    vue前端項目打包成Docker鏡像并運(yùn)行的實現(xiàn)

    這篇文章主要介紹了vue前端項目打包成Docker鏡像并運(yùn)行的實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08

最新評論