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


Posted in Javascript onMay 30, 2018

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

参考版本 vue源码版本:0.11

相关

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被访问

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

Javascript 相关文章推荐
比较简单的异步加载JS文件的代码
Jul 18 Javascript
jQuery给多个不同元素添加class样式的方法
Mar 26 Javascript
jquery控制表单输入框显示默认值的方法
May 22 Javascript
Bootstrap被封装的弹层
Jul 20 Javascript
js获取元素的标签名实现方法
Oct 08 Javascript
深究AngularJS中$sce的使用
Jun 12 Javascript
本地存储localStorage用法详解
Jul 31 Javascript
vuejs使用$emit和$on进行组件之间的传值的示例
Oct 04 Javascript
javascript Function函数理解与实战
Dec 01 Javascript
vue-router权限控制(简单方式)
Oct 29 Javascript
antd design table更改某行数据的样式操作
Oct 31 Javascript
js删除指定位置超链接中含有百度与360的标题
Jan 06 Javascript
Angular搜索场景中使用rxjs的操作符处理思路
May 30 #Javascript
微信小程序实现跑马灯效果完整代码(附效果图)
May 30 #Javascript
vue通过点击事件读取音频文件的方法
May 30 #Javascript
vue 表单输入格式化中文输入法异常问题
May 30 #Javascript
详解如何使用babel进行es6文件的编译
May 29 #Javascript
基于打包工具Webpack进行项目开发实例
May 29 #Javascript
JavaScript反射与依赖注入实例详解
May 29 #Javascript
You might like
php echo 输出字符串函数详解
2010/05/13 PHP
php处理文件的小例子(解压缩,删除目录)
2013/02/03 PHP
在smarty中调用php内置函数的方法
2013/02/07 PHP
PHP指定截取字符串中的中英文或数字字符的实例分享
2016/03/18 PHP
PHP实现的限制IP投票程序IP来源分析
2016/05/04 PHP
php事件驱动化设计详解
2016/11/10 PHP
JQuery实现简单的图片滑动切换特效
2015/11/22 Javascript
完美解决JS文件页面加载时的阻塞问题
2016/12/18 Javascript
使用nodeJs来安装less及编译less文件为css文件的方法
2017/11/20 NodeJs
vue实现的组件兄弟间通信功能示例
2018/12/04 Javascript
新版小程序登录授权的方法
2018/12/12 Javascript
layer 刷新某个页面的实现方法
2019/09/05 Javascript
kafka调试中遇到Connection to node -1 could not be established. Broker may not be available.
2019/09/17 Javascript
微信小程序实现比较功能的方法汇总(五种方法)
2020/03/07 Javascript
[12:51]71泪洒现场!是DOTA2让经典重现
2014/03/24 DOTA
python3库numpy数组属性的查看方法
2018/04/17 Python
PyQt5每天必学之布局管理
2018/04/19 Python
Python爬虫框架Scrapy基本用法入门教程
2018/07/26 Python
详解django实现自定义manage命令的扩展
2019/08/13 Python
python向图片里添加文字
2019/11/26 Python
Python模块_PyLibTiff读取tif文件的实例
2020/01/13 Python
python主要用于哪些方向
2020/07/05 Python
python实现测试工具(二)——简单的ui测试工具
2020/10/19 Python
Python基于内置函数type创建新类型
2020/10/22 Python
Python中读取文件名中的数字的实例详解
2020/12/25 Python
Kathmandu澳洲户外商店:新西兰户外运动品牌
2017/11/12 全球购物
List, Set, Map是否继承自Collection接口?
2016/05/16 面试题
Ajxa常见问题都有哪些
2014/03/26 面试题
php优化查询foreach代码实例讲解
2021/03/24 PHP
商业街策划方案
2014/05/31 职场文书
音乐会主持人开场白
2015/05/28 职场文书
行政复议决定书
2015/06/24 职场文书
党员心得体会范文2016
2016/01/23 职场文书
小学生作文写作技巧100例,非常实用!
2019/07/08 职场文书
java基础——多线程
2021/07/03 Java/Android
python基础之文件操作
2021/10/24 Python