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 相关文章推荐
javascript 表单的友好用户体现
Jan 07 Javascript
传智播客学习之java 反射
Nov 22 Javascript
jquery实现的让超出显示范围外的导航自动固定屏幕最顶上
Sep 22 Javascript
使用jquery读取html5 localstorage的值的方法
Jan 04 Javascript
用js写了一个类似php的print_r输出换行功能
Feb 18 Javascript
jquery 提交值不为空的元素示例代码
May 10 Javascript
js制作带有遮罩弹出层实现登录注册表单特效代码分享
Sep 05 Javascript
使用ionic切换页面卡顿的解决方法
Dec 16 Javascript
微信小程序开发之入门实例教程篇
Mar 07 Javascript
jQuery实现的点击标题文字切换字体效果示例【测试可用】
Apr 26 jQuery
Vue两种组件类型:递归组件和动态组件的用法
Aug 06 Javascript
vue项目实现减少app.js和vender.js的体积操作
Nov 12 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
基于PHP实现等比压缩图片大小
2016/03/04 PHP
php封装的数据库函数与用法示例【参考thinkPHP】
2016/11/08 PHP
PHP中功能强大却很少使用的函数实例小结
2016/11/10 PHP
(function($){...})(jQuery)的意思
2010/07/22 Javascript
JQUERY 实现窗口滚动搜索框停靠效果(类似滚动停靠)
2013/03/27 Javascript
为指定的元素添加遮罩层的示例代码
2014/01/15 Javascript
JavaScript简单实现弹出拖拽窗口(一)
2016/06/17 Javascript
react-native组件中NavigatorIOS和ListView结合使用的方法
2017/09/30 Javascript
VueCli3构建TS项目的方法步骤
2018/11/07 Javascript
JS中的算法与数据结构之栈(Stack)实例详解
2019/08/20 Javascript
JQuery通过键盘控制键盘按下与松开触发事件
2020/08/07 jQuery
[02:56]《DAC最前线》之国外战队抵达上海备战亚洲邀请赛
2015/01/28 DOTA
详解Python中使用base64模块来处理base64编码的方法
2016/07/01 Python
机器学习python实战之决策树
2017/11/01 Python
Python获取本机所有网卡ip,掩码和广播地址实例代码
2018/01/22 Python
python selenium 对浏览器标签页进行关闭和切换的方法
2018/05/21 Python
对python读取zip压缩文件里面的csv数据实例详解
2019/02/08 Python
python提取照片坐标信息的实例代码
2019/08/14 Python
python实现在线翻译功能
2020/03/03 Python
django的模型类管理器——数据库操作的封装详解
2020/04/01 Python
CSS3——齿轮转动关键代码
2013/05/02 HTML / CSS
css3实现平移效果(transfrom:translate)的示例
2020/11/13 HTML / CSS
联强国际笔试题面试题
2013/07/10 面试题
关于廉洁的广播稿
2014/01/30 职场文书
安全协议书
2014/04/23 职场文书
加入学生会演讲稿
2014/04/24 职场文书
环保项目建议书
2014/08/26 职场文书
房产授权委托书范本
2014/09/22 职场文书
出售房屋委托书范本
2014/09/24 职场文书
机票销售员态度不好检讨书
2014/09/27 职场文书
听证会主持词
2015/07/03 职场文书
家长会后的感想
2015/08/11 职场文书
教你快速开启Apache SkyWalking的自监控
2021/04/25 Servers
vue-cli4.5.x快速搭建项目
2021/05/30 Vue.js
浅谈react useEffect闭包的坑
2021/06/08 Javascript
继承Win10缺点!教你关闭Win11烦人的网络搜索
2021/11/23 数码科技