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

vue源碼學習之Object.defineProperty 對數(shù)組監(jiān)聽

 更新時間:2018年05月30日 09:50:54   作者:iiijarvis  
這篇文章主要介紹了vue源碼學習之Object.defineProperty 對數(shù)組監(jiān)聽,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

上一篇中,我們介紹了一下defineProperty 對對象的監(jiān)聽,這一篇我們看下defineProperty 對數(shù)組的監(jiān)聽

數(shù)組的變化

先讓我們了解下Object.defineProperty()對數(shù)組變化的跟蹤情況:

var a={};
bValue=1;
Object.defineProperty(a,"b",{
  set:function(value){
    bValue=value;
    console.log("setted");
  },
  get:function(){
    return bValue;
  }
});
a.b;//1
a.b=[];//setted
a.b=[1,2,3];//setted
a.b[1]=10;//無輸出
a.b.push(4);//無輸出
a.b.length=5;//無輸出
a.b;//[1,10,3,4,undefined];

可以看到,當a.b被設(shè)置為數(shù)組后,只要不是重新賦值一個新的數(shù)組對象,任何對數(shù)組內(nèi)部的修改都不會觸發(fā)setter方法的執(zhí)行。這一點非常重要,因為基于Object.defineProperty()方法的現(xiàn)代前端框架實現(xiàn)的數(shù)據(jù)雙向綁定也同樣無法識別這樣的數(shù)組變化。因此第一點,如果想要觸發(fā)數(shù)據(jù)雙向綁定,我們不要使用arr[1]=newValue;這樣的語句來實現(xiàn);第二點,框架也提供了許多方法來實現(xiàn)數(shù)組的雙向綁定。

對于框架如何實現(xiàn)數(shù)組變化的監(jiān)測,大多數(shù)情況下,框架會重寫Array.prototype.push方法,并生成一個新的數(shù)組賦值給數(shù)據(jù),這樣數(shù)據(jù)雙向綁定就會觸發(fā)。

實現(xiàn)簡單的對數(shù)組的變化的監(jiān)聽

var arrayPush = {};
(function(method){
  var original = Array.prototype[method];
  arrayPush[method] = function() {
    // this 指向可通過下面的測試看出
    console.log(this);
    return original.apply(this, arguments)
  };
})('push');

var testPush = [];
testPush.__proto__ = arrayPush;
// 通過輸出,可以看出上面所述 this 指向的是 testPush
// []
testPush.push(1);
// [1]
testPush.push(2);

在官方文檔,所需監(jiān)視的只有 push()、pop()、shift()、unshift()、splice()、sort()、reverse() 7 種方法。我們可以遍歷一下:

var arrayProto = Array.prototype
var arrayMethods = Object.create(arrayProto)

;[
 'push',
 'pop',
 'shift',
 'unshift',
 'splice',
 'sort',
 'reverse'
].forEach(function(item){
  Object.defineProperty(arrayMethods,item,{
    value:function mutator(){
      //緩存原生方法,之后調(diào)用
      console.log('array被訪問');
      var original = arrayProto[item]  
      var args = Array.from(arguments)
    original.apply(this,args)
      // console.log(this);
    },
  })
})

完整代碼

function Observer(data){
  this.data = data;
  this.walk(data);
}

var p = Observer.prototype;
var arrayProto = Array.prototype
var arrayMethods = Object.create(arrayProto)

;[
 'push',
 'pop',
 'shift',
 'unshift',
 'splice',
 'sort',
 'reverse'
].forEach(function(item){
  Object.defineProperty(arrayMethods,item,{
    value:function mutator(){
      //緩存原生方法,之后調(diào)用
      console.log('array被訪問');
      var original = arrayProto[item]  
      var args = Array.from(arguments)
    original.apply(this,args)
      // console.log(this);
    },
  })
})

p.walk = function(obj){
  var value;
  for(var key in obj){
    // 通過 hasOwnProperty 過濾掉一個對象本身擁有的屬性 
    if(obj.hasOwnProperty(key)){
      value = obj[key];
      // 遞歸調(diào)用 循環(huán)所有對象出來
      if(typeof value === 'object'){
        if (Array.isArray(value)) {
          var augment = value.__proto__ ? protoAugment : copyAugment 
          augment(value, arrayMethods, key)
          observeArray(value)
        }
        new Observer(value);
      }
      this.convert(key, value);
    }
  }
};

p.convert = function(key, value){
  Object.defineProperty(this.data, key, {
    enumerable: true,
    configurable: true,
    get: function(){
      console.log(key + '被訪問');
      return value;
    },
    set: function(newVal){
      console.log(key + '被修改,新' + key + '=' + newVal);
      if(newVal === value) return ;
      value = newVal;
    }
  })
}; 

var data = {
  user: {
    // name: 'zhangsan',
    age: function(){console.log(1)}
  },
  apg: [{'a': 'b'},2,3]
}

function observeArray (items) {
  for (var i = 0, l = items.length; i < l; i++) {
    observe(items[i])
  }
}

//數(shù)據(jù)重復Observer
function observe(value){
  if(typeof(value) != 'object' ) return;
  var ob = new Observer(value)
   return ob;
}

//輔助方法
function def (obj, key, val) {
 Object.defineProperty(obj, key, {
  value: val,
  enumerable: true,
  writable: true,
  configurable: true
 })
}

// 兼容不支持__proto__的方法
//重新賦值A(chǔ)rray的__proto__屬性
function protoAugment (target,src) {
 target.__proto__ = src
}
//不支持__proto__的直接修改相關(guān)屬性方法
function copyAugment (target, src, keys) {
 for (var i = 0, l = keys.length; i < l; i++) {
  var key = keys[i]
  def(target, key, src[key])
 }
}

var app = new Observer(data);
// data.apg[2] = 111;
data.apg.push(5);
// data.apg[0].a = 10;
// console.log(data.apg);

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • vue3使用base64加密的兩種方法舉例

    vue3使用base64加密的兩種方法舉例

    這篇文章主要給大家介紹了關(guān)于vue3使用base64加密的兩種方法,我們在vue項目中有時會使用到Base6464轉(zhuǎn)碼,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2023-07-07
  • vue接入下載文件接口問題

    vue接入下載文件接口問題

    這篇文章主要介紹了vue接入下載文件接口問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • Vue3+Element?Plus按需引入(自動導入)詳解

    Vue3+Element?Plus按需引入(自動導入)詳解

    element-plus根據(jù)官網(wǎng)文檔,推薦用戶采用按需導入的方式進行導入,下面這篇文章主要給大家介紹了關(guān)于Vue3+Element?Plus按需引入(自動導入)的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-10-10
  • Vue組件實現(xiàn)旋轉(zhuǎn)木馬動畫

    Vue組件實現(xiàn)旋轉(zhuǎn)木馬動畫

    這篇文章主要為大家詳細介紹了Vue組件實現(xiàn)旋轉(zhuǎn)木馬動畫效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-07-07
  • Element?ui關(guān)閉el-dialog時如何清除數(shù)據(jù)

    Element?ui關(guān)閉el-dialog時如何清除數(shù)據(jù)

    這篇文章主要介紹了Element?ui關(guān)閉el-dialog時如何清除數(shù)據(jù)問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-02-02
  • vue實現(xiàn)搜索功能

    vue實現(xiàn)搜索功能

    這篇文章主要為大家詳細介紹了vue實現(xiàn)搜索功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-05-05
  • Vue.js slot插槽的作用域插槽用法詳解

    Vue.js slot插槽的作用域插槽用法詳解

    這篇文章主要介紹了Vue.js slot插槽的作用域插槽用法詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下
    2021-09-09
  • Vue3中Vite和Vue-cli的特點與區(qū)別詳解

    Vue3中Vite和Vue-cli的特點與區(qū)別詳解

    vue-cli是Vue早期推出的一款腳手架,使用webpack創(chuàng)建Vue項目,可以選擇安裝需要的各種插件,比如Vuex、VueRouter,下面這篇文章主要給大家介紹了關(guān)于Vue3中Vite和Vue-cli的特點與區(qū)別的相關(guān)資料,需要的朋友可以參考下
    2022-12-12
  • Vue 中使用vue2-highcharts實現(xiàn)top功能的示例

    Vue 中使用vue2-highcharts實現(xiàn)top功能的示例

    下面小編就為大家分享一篇Vue 中使用vue2-highcharts實現(xiàn)top功能的示例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-03-03
  • Backbone前端框架核心及源碼解析

    Backbone前端框架核心及源碼解析

    這篇文章主要為大家介紹了Backbone前端框架核心及源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-02-02

最新評論