Vue数据驱动模拟实现2


Posted in Javascript onJanuary 11, 2017

一、前言

在随笔“模拟Vue之数据驱动1”结尾处,我们说到如果监听的属性是个对象呢?那么这个对象中的其他属性岂不就是监听不了了吗?

如下:

 Vue数据驱动模拟实现2

倘若user中的name、age属性变化,如何知道它们变化了呢?

今儿,就来解决这一问题。

通过走读Vue源码,发现他是利用Observer构造函数为每个对象创建一个Observer对象,来监听数据的,如果数据中的属性又是一个对象,那么就又通过Observer来监听嘛。

其实,核心思想就是树的先序遍历(关于树,可参考here)。如我们将上述Demo中的data数据,图形化一下,就更加明白了,如下:

Vue数据驱动模拟实现2 

好了,理清了大体思路,下面我们就一起来创建一个Observer吧。

二、Observer构造

Observer整体结构如下:

function Observer(data){
 //如若this不是Observer对象,即创建一个
 if(!(this instanceof Observer)){
 return new Observer(data);
 }
 this.data = data;
 this.walk(data); 
}

let p = Observer.prototype = Object.create(null);

p.walk = function(data){
 /*
 TODO:监听data数据中的所有属性,
 并查看data中属性值是否为对象,
 若为对象,就创建一个Observer实例
 */ 
}

p.convert = function(key, val){
 //TODO:通过Object.defineProperty监听数据 
}

好了,下面,我们一起来完成walk以及convert方法吧。

-walk-

首先,我们在walk方法中实现对data对象中的所有属性监听,如下:

p.walk = function(data){
 let keys = Object.keys(data);
 keys.forEach( key => {
 let val = data[key];
 this.convert(key, val);
 });
}

且,由于属性中可能又会是一个对象,那么,我们就有必要监听它们。

怎么办呢?

如果是个对象,再次利用Observer构造函数,处理它不就完了么。

如下:

p.walk = function(data){
 let keys = Object.keys(data);
 keys.forEach( key => {
 let val = data[key];
 //如果val为对象,则交给Observer处理
 if(typeof val === 'object'){
  Observer(val);
 }
 this.convert(key, val);
 });
}

你可能会有这样的疑问,如果直接利用Observer处理对象,那么不就与父对象失去关联了么?

然而并没有,因为JavaScript对于对象是指向地址关系,所以怎么会失去关联呢。

-convert-

对于convert方法,就比较简单了,一如既往就是利用Object.defineProperty监听数据,如下:

p.convert = function(key, val){
 Object.defineProperty(this.data, key, {
 get: ()=>{
  console.log('访问了'+key+' 值为'+val);
  return val;
 },
 set: (newVal)=>{
  console.log('设置了'+key+' 值为'+newVal);
  if(newVal !== val){
  val = newVal;
  }
 }
 });
}

好了,到此,一个简单的Observer就构造完成,下面我们就来测试下,是否成功监听了每个属性。

<script src="./observer.js"></script>
<script>
 let data = {
 user: {
  name: 'Monkey',
  age: 24
 },
 lover: {
  name: 'Dorie',
  age: 23
 }
 };
 Observer(data);
</script>

效果如下:

Vue数据驱动模拟实现2

Perfect,完整代码见github。

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

Javascript 相关文章推荐
广泛收集的jQuery拖放插件集合
Apr 09 Javascript
JS使用eval解析JSON的注意事项分析
Nov 14 Javascript
基于jQuery的ajax方法封装
Jul 14 Javascript
详解jQuery的表单验证插件--Validation
Dec 21 Javascript
JS中的作用域链
Mar 01 Javascript
微信小程序页面跳转功能之从列表的item项跳转到下一个页面的方法
Nov 27 Javascript
Vue官网todoMVC示例代码
Jan 29 Javascript
vue watch深度监听对象实现数据联动效果
Aug 16 Javascript
原生JS实现动态添加新元素、删除元素方法
May 05 Javascript
vue日历/日程提醒/html5本地缓存功能
Sep 02 Javascript
jQuery实现全选、反选和不选功能的方法详解
Dec 04 jQuery
JavaScript使用canvas绘制随机验证码
Feb 17 Javascript
jQuery实现对象转为url参数的方法
Jan 11 #Javascript
将鼠标焦点定位到文本框最后(代码分享)
Jan 11 #Javascript
移动端界面的适配
Jan 11 #Javascript
bootstrapValidator bootstrap-select验证不可用的解决办法
Jan 11 #Javascript
HTML5 js实现拖拉上传文件功能
Nov 20 #Javascript
Bootstrap表单控件使用方法详解
Jan 11 #Javascript
老生常谈的跨域处理
Jan 11 #Javascript
You might like
不用数据库的多用户文件自由上传投票系统(1)
2006/10/09 PHP
PHP实现多服务器session共享之NFS共享的方法
2007/03/16 PHP
PHP中获取文件扩展名的N种方法小结
2012/02/27 PHP
php 的加密函数 md5,crypt,base64_encode 等使用介绍
2012/04/09 PHP
php获取一个变量的名字的方法
2014/09/05 PHP
php模拟用户自动在qq空间发表文章的方法
2015/01/07 PHP
Laravel框架表单验证操作实例分析
2019/09/30 PHP
Javascript和Ajax中文乱码吐血版解决方案
2009/12/21 Javascript
jquery ajax的success回调函数中实现按钮置灰倒计时
2013/11/19 Javascript
JavaScript中的getDay()方法使用详解
2015/06/09 Javascript
使用bootstrap3开发响应式网站
2016/05/12 Javascript
Bootstrap学习笔记之js组件(4)
2016/06/12 Javascript
聊一聊JS中this的指向问题
2016/06/17 Javascript
AngularJS递归指令实现Tree View效果示例
2016/11/07 Javascript
使用base64对图片的二进制进行编码并用ajax进行显示
2017/01/03 Javascript
vue.js+Element实现表格里的增删改查
2017/01/18 Javascript
捕获未处理的Promise错误方法
2017/10/13 Javascript
浅析Vue自定义组件的v-model
2017/11/26 Javascript
Vue 实现输入框新增搜索历史记录功能
2019/10/15 Javascript
jQuery中event.target和this的区别详解
2020/08/13 jQuery
[13:18]《一刀刀一天》之DOTA全时刻21:详解TI新赛制 A队再露獠牙
2014/06/24 DOTA
[05:28]刀塔密之一:团结则存
2014/07/03 DOTA
[08:53]DOTA2-DPC中国联赛 正赛 PSG.LGD vs LBZS 选手采访
2021/03/11 DOTA
Python 字符串操作实现代码(截取/替换/查找/分割)
2013/06/08 Python
python实现网站的模拟登录
2016/01/04 Python
python 队列详解及实例代码
2016/10/18 Python
Python爬虫包BeautifulSoup学习实例(五)
2018/06/17 Python
Python matplotlib以日期为x轴作图代码实例
2019/11/22 Python
解决python pandas读取excel中多个不同sheet表格存在的问题
2020/07/14 Python
求职自荐信范文(优秀篇)
2015/03/27 职场文书
公司搬迁通知
2015/04/20 职场文书
mysql 8.0.24 安装配置方法图文教程
2021/05/12 MySQL
Android超详细讲解组件ScrollView的使用
2022/03/31 Java/Android
pycharm无法安装cv2模块问题
2022/05/20 Python
Linux服务器离线安装 nginx的详细步骤
2022/06/16 Servers
Pygame游戏开发之太空射击实战敌人精灵篇
2022/08/05 Python