3分钟了解vue数据劫持的原理实现


Posted in Javascript onMay 01, 2019

目的: 了解Object.defineProperty如何实现数据劫持

大致原理是这样的:

  1. 定义一个监听函数,对对象的每一个属性进行监听
  2. 通过Object.defineProperty对监听的每一个属性设置get 和 set 方法。
  3. 对对象实行监听
  4. 对对象内嵌对象进行处理
  5. 对数组对象进行处理

1. 先定义一个对象

let obj = {
 name: 'jw'
}

2. 定义一个监听函数

/**
* 判断监听的是否是对象
* 如果是对象,就遍历,并且对每个属性进行定义get 和 set
*/

function observer(obj) {
 if(typeof obj === 'object') {
  for (let key in obj) {
  // defineReactive 方法设置get和set,见第三步
   defineReactive(obj, key, obj[key]);
  }
 }
}

3.定义一个函数,处理每个属性

function defineReactive(obj, key, value) {
 Object.defineProperty(obj, key, {
  get() {
   return value;
  },
  set(val) {
   console.log('数据更新了')
   value = val;
  }
 })
}

ok. 到这里初版已经实现了。尝试一下吧

observer(obj);
obj.name = 'haha'

控制台输出:
//数据更新了

以上已经实现设置obj的属性的时候,被监听到,并且可以去执行一些代码了。但是,如果对象里面嵌入了对象呢?比如:

let obj = {
 name: 'jw',
 age: {
  old: 60
 }
}

执行以下代码

observer(obj);
obj.age.old = '50'

控制台输出: 空

4.对监控的obj进行迭代处理

// 修改defineReactive , 添加一行代码
function defineReactive(obj, key, value) {
 // 如果对象的属性也是一个对象。迭代处理
 observer(value);
 Object.defineProperty(obj, key, {
  //....
 })
}

再执行以下代码:

observer(obj);
obj.age.old = '50'

控制台输出:
//数据更新了

可惜的是,如果对象是一个数组,Object.defineProperty就无法起作用了,比如:

obj.skill = [1, 2, 3];
obj.age.push(4);

控制台输出:
//空

实际上,不止push,包括slice,shift,unshif...都是没有作用.

5. 重写数组的方法

let arr = ['push', 'slice', 'shift', 'unshift'];
arr.forEach(method=> {
 let oldPush = Array.prototype[method];
 Array.prototype[method] = function(value) {
  console.log('数据更新了')
  oldPush.call(this, value)
 }
})

再执行以下代码:

obj.skill = [1, 2, 3];
obj.skill.push(4);

控制台输出:
//数据更新了

但是,数组的length操作仍然是无效的。这也是为什么vue中只能通过方法去改变数组的原因了。

总结: Object.defineProperty只是解决了状态变更后,如何触发通知的问题,那要通知谁呢?谁会关心那些属性发生了变化呢?以后再说。

以下完整代码

let obj = {
 name: 'jw',
 age: {
  old: '60'
 }
}

// vue 数据劫持 Observer.defineProperty

function observer(obj) {
 if(typeof obj === 'object') {
  for (let key in obj) {
   defineReactive(obj, key, obj[key]);
  }
 }
}

function defineReactive(obj, key, value) {
 observer(value);

 Object.defineProperty(obj, key, {
  get() {
   return value;
  },
  set(val) {
   console.log('数据更新了')
   value = val;
  }
 })
}
observer(obj);


// obj.age.old = '50'


// Object.defineProperty 对 数组无效
let arr = ['push', 'slice', 'shift', 'unshift'];

arr.forEach(method=> {
 let oldPush = Array.prototype[method];
 Array.prototype[method] = function(value) {
  console.log('数据更新了')
  oldPush.call(this, value)
 }
})
obj.skill = [1, 2, 3];
obj.skill.push(4);

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

Javascript 相关文章推荐
javascript 数组学习资料收集
Apr 11 Javascript
js操作时间(年-月-日 时-分-秒 星期几)
Jun 20 Javascript
Javascript 检测键盘按键信息及键码值对应介绍
Jan 03 Javascript
jQuery如何将选中的对象转化为原始的DOM对象
Jun 09 Javascript
hovertree插件实现二级树形菜单(简单实用)
Dec 28 Javascript
详解AngularJs路由之Ui-router-resolve(预加载)
Jun 13 Javascript
如何将 jQuery 从你的 Bootstrap 项目中移除(取而代之使用Vue.js)
Jul 17 jQuery
详解React-Native全球化多语言切换工具库react-native-i18n
Nov 03 Javascript
React Native 自定义下拉刷新上拉加载的列表的示例
Mar 01 Javascript
vuex 项目结构目录及一些简单配置介绍
Apr 08 Javascript
angular 未登录状态拦截路由跳转的方法
Oct 09 Javascript
layui中的switch开关实现方法
Sep 03 Javascript
vue 对象添加或删除成员时无法实时更新的解决方法
May 01 #Javascript
JavaScript强制类型转换和隐式类型转换操作示例
May 01 #Javascript
Vue源码之关于vm.$delete()/Vue.use()内部原理详解
May 01 #Javascript
Vue.extend实现挂载到实例上的方法
May 01 #Javascript
JS html事件冒泡和事件捕获操作示例
May 01 #Javascript
JS实现的贪吃蛇游戏案例详解
May 01 #Javascript
javascript原型链学习记录之继承实现方式分析
May 01 #Javascript
You might like
通过PHP修改Linux或Unix口令的方法分享
2012/01/30 PHP
php中使用in_array() foreach array_search() 查找数组是否包含时的性能对比
2015/04/14 PHP
PHP通过串口实现发送短信
2015/07/08 PHP
php如何把表单内容提交到数据库
2019/07/08 PHP
select组合框option的捕捉实例代码
2008/09/30 Javascript
javascript-简单的计算器实现步骤分解(附图)
2013/05/30 Javascript
jquery实现图片滚动效果的简单实例
2013/11/23 Javascript
jQuery中end()方法用法实例
2015/01/08 Javascript
jQuery检测输入的字符串包含的中英文的数量
2015/04/17 Javascript
AngularJS中一般函数参数传递用法分析
2016/11/22 Javascript
jQuery插件echarts实现的多柱子柱状图效果示例【附demo源码下载】
2017/03/04 Javascript
JQuery Ajax 异步操作之动态添加节点功能
2017/05/24 jQuery
bootstrap fileinput实现文件上传功能
2017/08/23 Javascript
Vue实例中生命周期created和mounted的区别详解
2017/08/25 Javascript
Vuex,iView UI面包屑导航使用扩展详解
2019/11/04 Javascript
node.JS事件机制与events事件模块的使用方法详解
2020/02/06 Javascript
Python简单连接MongoDB数据库的方法
2016/03/15 Python
深入理解python中的select模块
2017/04/23 Python
python 读取视频,处理后,实时计算帧数fps的方法
2018/07/10 Python
python使用webdriver爬取微信公众号
2018/08/31 Python
Python实现的字典排序操作示例【按键名key与键值value排序】
2018/12/21 Python
python 使用pygame工具包实现贪吃蛇游戏(多彩版)
2019/10/30 Python
pytorch实现对输入超过三通道的数据进行训练
2020/01/15 Python
keras 获取某层输出 获取复用层的多次输出实例
2020/05/23 Python
Python Merge函数原理及用法解析
2020/09/16 Python
浅谈HTML5中dialog元素尝鲜
2018/10/15 HTML / CSS
阿玛瑞酒店中文官方网站:Amari.com
2018/02/13 全球购物
Geekbuying波兰:购买中国电子产品
2019/10/20 全球购物
八一建军节部队活动方案
2014/02/04 职场文书
基层党组织建设整改方案
2014/09/16 职场文书
农村党支部承诺书
2015/04/30 职场文书
公司的力量观后感
2015/06/05 职场文书
《狼王梦》读后感:可怜天下父母心
2019/11/01 职场文书
节约用水广告语60条
2019/11/14 职场文书
Nginx如何配置Http、Https、WS、WSS的方法步骤
2021/05/11 Servers
Python 中的单分派泛函数你真的了解吗
2021/06/22 Python