欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

vue源码学习之Object.defineProperty对象属性监听

程序员文章站 2022-05-25 22:23:22
本文介绍了vue源码学习之object.defineproperty对象属性监听,分享给大家,具体如下: 参考版本 相关 vue实现双向数据绑定的关键是 obj...

本文介绍了vue源码学习之object.defineproperty对象属性监听,分享给大家,具体如下:

参考版本

相关

vue实现双向数据绑定的关键是 object.defineproperty ,让我们先来看下这个函数。

在mdn上查看有关object.defineproperty 的解释。

我们先从最简单的开始:

let a = {'b': 1};
object.defineproperty(a, 'b', {
  enumerable: false,
  configurable: false,
  get: function(){
    console.log('b' + '被访问');
  },
  set: function(newval){
    console.log('b' + '被修改,新' + 'b' + '=' + newval);
  }
});

a.b = 2;  // b被修改,新b=2
a.b;    // b被访问

这样,我们就能监听对象了!但问题并不仅仅这么简单。。。

我们可能会有对象中属性的值还是对象这种嵌套情况,可以通过递归解决!

在vue源代码文件 srcobserveobserver.js 中

// 观察者构造函数
function observer(data){
  this.data = data;
  this.walk(data);
}

let p = observer.prototype;
p.walk = function(obj){
  let val;
  for(let key in obj){
    // 通过 hasownproperty 过滤掉一个对象本身拥有的属性 
    if(obj.hasownproperty(key)){
      val = obj[key];
      // 递归调用 循环所有对象出来
      if(typeof val === 'object'){
        new observer(val);
      }
      this.convert(key, val);
    }
  }
};

p.convert = function(key, val){
  object.defineproperty(this.data, key, {
    enumerable: false,
    configurable: false,
    get: function(){
      console.log(key + '被访问');
    },
    set: function(newval){
      console.log(key + '被修改,新' + key + '=' + newval);
      if(newval === val) return ;
      val = newval;
    }
  })
};

let data = {
  user: {
    name: 'zhangsan',
    age: 14
  },
  address: {
    city: 'beijing'
  }
}

let app = new observer(data);
data.user.name;  // user被访问 

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。