2019独角兽企业重金招聘Python工程师标准>>>

首先 Object.defineProperty(obj,prop,descriptor) 用法介绍:

  • 参数

    obj:目标对象

    prop:需要定义的属性或方法的名称

    descriptor:目标属性所拥有的特性

  • 可供定义的特性列表

    value:属性的值

    writable:如果为false,属性的值就不能被重写。

    get: 一旦目标属性被访问就会调回此方法,并将此方法的运算结果返回用户。

    set:一旦目标属性被赋值,就会调回此方法。

    configurable:如果为false,则任何尝试删除目标属性或修改属性性以下特性(writable, configurable, enumerable)的行为将被无效化。

    enumerable:是否能在for...in循环中遍历出来或在Object.keys中列举出来

什么是数据劫持

通过上面对Object.defineProperty的介绍,我们不难发现,当我们访问或设置对象的属性的时候,都会触发相对应的函数,然后在这个函数里返回或设置属性的值。既然如此,我们当然可以在触发函数的时候动一些手脚做点我们自己想做的事情,这也就是“劫持”操作。在Vue中其实就是通过Object.defineProperty来劫持对象属性的setter和getter操作,并“种下”一个监听器,当数据发生变化的时候发出通知。先简单的举个例子:

var data = {name:'fang'
}Object.keys(data).forEach(function(key){Object.defineProperty(data,key,{enumerable:true,configurable:true,get:function(){console.log('get');},set:function(){console.log('监听到数据发生了变化');}})
});
data.name //控制台会打印出 “get”
data.name = 'fangpeng' //控制台会打印出 "监听到数据发生了变化"

vue原理:

监听对象属性的变化

Vue数据绑定之前,它通过observe每个对象的属性,添加到订阅器dep中,当数据发生变化的时候发出一个notice。 相关源代码如下:(作者采用的是ES6+flow写的,代码在src/core/observer/index.js模块里面)

export function defineReactive (obj: Object,key: string,val: any,customSetter?: Function
) {const dep = new Dep()//创建订阅对象const property = Object.getOwnPropertyDescriptor(obj, key)//获取obj对象的key属性的描述//属性的描述特性里面如果configurable为false则属性的任何修改将无效if (property && property.configurable === false) {return}// cater for pre-defined getter/settersconst getter = property && property.getconst setter = property && property.setlet childOb = observe(val)//创建一个观察者对象Object.defineProperty(obj, key, {enumerable: true,//可枚举configurable: true,//可修改get: function reactiveGetter () {const value = getter ? getter.call(obj) : val//先调用默认的get方法取值//这里就劫持了get方法,也是作者一个巧妙设计,在创建watcher实例的时候,通过调用对象的get方法往订阅器dep上添加这个创建的watcher实例if (Dep.target) {dep.depend()if (childOb) {childOb.dep.depend()}if (Array.isArray(value)) {dependArray(value)}}return value//返回属性值},set: function reactiveSetter (newVal) {const value = getter ? getter.call(obj) : val//先取旧值if (newVal === value) {return}//这个是用来判断生产环境的,可以无视if (process.env.NODE_ENV !== 'production' && customSetter) {customSetter()}if (setter) {setter.call(obj, newVal)} else {val = newVal}childOb = observe(newVal)//继续监听新的属性值dep.notify()//这个是真正劫持的目的,要对订阅者发通知了}})
}

以上是Vue监听对象属性的变化,那么问题来了,我们经常在传递数据的时候往往不是一个对象,很有可能是一个数组,那是不是就没有办法了呢,答案显然是否则的。那么下面就看看作者是如何监听数组的变化:

监听数组的变化

我们还看先看这段源码:

const arrayProto = Array.prototype//原生Array的原型
export const arrayMethods = Object.create(arrayProto);['push','pop','shift','unshift','splice','sort','reverse'
]
.forEach(function (method) {const original = arrayProto[method]//缓存元素数组原型//这里重写了数组的几个原型方法def(arrayMethods, method, function mutator () {//这里备份一份参数应该是从性能方面的考虑let i = arguments.lengthconst args = new Array(i)while (i--) {args[i] = arguments[i]}const result = original.apply(this, args)//原始方法求值const ob = this.__ob__//这里this.__ob__指向的是数据的Observerlet insertedswitch (method) {case 'push':inserted = argsbreakcase 'unshift':inserted = argsbreakcase 'splice':inserted = args.slice(2)break}if (inserted) ob.observeArray(inserted)// notify changeob.dep.notify()return result})
})...
//定义属性
function def (obj, key, val, enumerable) {Object.defineProperty(obj, key, {value: val,enumerable: !!enumerable,writable: true,configurable: true});
}

上面的代码主要是继承了Array本身的原型方法,然后又做了劫持修改,可以发出通知。Vue在observer数据阶段会判断如果是数组的话,则修改数组的原型,这样的话,后面对数组的任何操作都可以在劫持的过程中控制。结合Vue的思想,我简单的写个小demo方便更好的理解:

var arrayMethod = Object.create(Array.prototype);
['push','shift'].forEach(function(method){Object.defineProperty(arrayMethod,method,{value:function(){var i = arguments.lengthvar args = new Array(i)while (i--) {args[i] = arguments[i]}var original = Array.prototype[method];var result = original.apply(this,args);console.log("已经控制了,哈哈");return result;},enumerable: true,writable: true,configurable: true})
})
var bar = [1,2];
bar.__proto__ = arrayMethod;
bar.push(3);//控制台会打印出 “已经控制了,哈哈”;并且bar里面已经成功的添加了成员 ‘3’ 

整个过程看起来好像没有什么问题,似乎Vue已经做到了完美,其实不然,Vue还是不能检测到数据项和数组长度改变的变化,例如下面的调用:

vm.items[index] = "xxx";
vm.items.length = 100;

我们尽量避免这样的调用方式,如果确实需要,作者也帮我们实现了一个$set操作,这里就不做介绍了。

实现对象属性代理

正常情况下我们是这样实例化一个Vue对象:

var VM = new Vue({data:{name:'lhl'},el:'#id'
})

按理说我们操作数据的时候应该是VM.data.name = ‘hxx’才对,但是作者觉得这样不够简洁,所以又通过代理的方式实现了VM.name = ‘hxx’的可能。 相关代码如下:

function proxy (vm, key) {if (!isReserved(key)) {Object.defineProperty(vm, key, {configurable: true,enumerable: true,get: function proxyGetter () {return vm._data[key]},set: function proxySetter (val) {vm._data[key] = val;}});}
}

表面上看起来我们是在操作VM.name,实际上还是通过Object.defineProperty()中的get和set方法劫持实现的。

总结

Vue框架很好的利用了Object.defineProperty()这个方法来实现了数据的双向绑定,同时也达到了很好的模块间解耦

转载于:https://my.oschina.net/pengpengpengone/blog/1837257

vue 数据劫持详解相关推荐

  1. 数据劫持详解(JavaScript)

    什么是数据劫持? 定义 数据劫持,其实就是数据代理. 数据劫持,指的是在访问或者修改对象的某个属性时,通过一段代码拦截这个行为,进行额外的操作或者修改返回结果. Demo 使用Object.defin ...

  2. 一篇由简到难的 Vue 面试题+详解答案

    作者:Big shark@LX 原文:https://juejin.cn/post/6961222829979697165 前言 本文整理了「高频」出现的 Vue 相关面试题并且附带「详解答案」 难度 ...

  3. 最全最详细前端vue面试题+详解答案(拿到高薪offer不是梦)

    最全vue面试题+详解答案 1.MVC 和 MVVM 区别 MVC MVC全名是 Model View Controller,时模型 - 视图 - 控制器的缩写,一种软件设计典范. Model(模型) ...

  4. 【面试题】973- 一篇由简到难的 Vue 面试题+详解答案

    作者:Big shark@LX 原文:https://juejin.cn/post/6961222829979697165 前言 本文整理了「高频」出现的 Vue 相关面试题并且附带「详解答案」 难度 ...

  5. 最全的 Vue 面试题+详解答案

    前言 本文整理了高频出现的 Vue 相关面试题并且附带详解答案 难度分为简单 中等 困难 三种类型 大家可以先不看答案自测一下自己的 Vue 水平哈 如果对原理感兴趣的同学 欢迎查看小编的手写 Vue ...

  6. vue组件component详解

    vue组件component详解 组件介绍 组件: ​ 组件是拥有一定功能多个html标签的集合体,是对html标签的封装. 好处: ​ 模板中为了实现一个(例如分页)效果,需要绘制20个html标签 ...

  7. vue 源码详解(零):Vue 源码流程图

    vue 源码详解(零):Vue 源码流程图 最近在研究 Vue 的源码, 整理博客, 结果想到的.看到的内容实在是太多了, 不知道从何写起, 故整理了一个大致的流程图,根据这个顺序进行一一整理. 为了 ...

  8. 每日一练:Python爬虫爬取全国新冠肺炎疫情数据实例详解,使用beautifulsoup4库实现

    Python 爬虫篇 - 爬取全国新冠肺炎疫情数据实例详解 效果图展示 第一章:疫情信息的下载与数据提取 ① 爬取页面数据到本地 ② json 字符串正则表达式分析 ③ 提取数据中的 json 字符串 ...

  9. 稳扎稳打Silverlight(17) - 2.0数据之详解DataGrid, 绑定数据到ListBox

    [索引页] [源码下载] 稳扎稳打Silverlight(17) - 2.0数据之详解DataGrid, 详解ListBox 作者:webabcd 介绍 Silverlight 2.0 详解DataG ...

最新文章

  1. Expo 快速开启之旅
  2. javaweb:session
  3. 组装自己的tesla超级计算机
  4. 滚动条组件 http://www.w3cfuns.com/notes/15098/96195b77bdbcb601590f67f971770bb8.html
  5. 数据分析必备的统计学(二):假设检验
  6. Atitit 数据挖掘技术体系 目录 1. 统计分析(分组聚合等 1 2. Tag标注 结构化 1 2.1. · 复杂数据类型挖掘(Text, Web 2 2.2. ,图形图像,视频,音频等) 2
  7. 硬盘服务器作用,文件服务器有什么作用?
  8. 浅谈Java及应用学java
  9. 计算机控制系统模型,计算机控制系统的数学模型.ppt
  10. 无人机欧拉角万向节锁死详解
  11. TeXmacs 中文输入问题
  12. cas:1092775-62-6 ; (ir[dfcf3ppy]2(bpy))pf6热延迟荧光材料TADF
  13. 王安计算机科学思想,【OHI访谈手记】互联网口述历史访谈计算机先驱John E. Savage...
  14. 双堆1.数据流的中位数
  15. 说一说Qpython在Android手机上的应用(二)
  16. php phpexcel 内容靠最右边_PHPExcel的使用小结,不够全面,只是应对当前我用到的地方做了笔记,日后再有新的需求还会随时修改...
  17. 为什么说速度超过光速能回到过去
  18. 关于谢尔宾斯基三角(Sierpinski)的讲解
  19. 中国网络安全人才需求
  20. 衡水二中2021年高考成绩查询,2020衡水二中高考喜报

热门文章

  1. HP DL380 G6安装Windows server 2003(有光驱和无光驱两种方法)
  2. 用yui compressor 压缩 javascirpt脚本
  3. 技术开发项目收获了什么_有幸入职字节跳动一年,附上百度、字节跳动等后台面经,谈谈我的收获!...
  4. ios uilabel 字体加粗
  5. IOS日历显示12个月
  6. java中可以改变引用的指向吗_java中引用传递问题,在函数中修改引用的指向,会不会影响实参?...
  7. SQLite中的运算符表达式
  8. WordPress漏洞扫描工具WPScan
  9. 网络协议图形化分析工具EtherApe
  10. John the Ripper