vue2.x是利用Object.defineProperty劫持对象或对象的属性(指的是在访问或者修改对象的某个属性时,通过一段代码拦截这个行为,进行额外的操作或者修改返回结果)的访问器,在属性值发生变化时获取属性值变化, 从而进行后续操作。
数据代理的的demo(通过一个对象代理对另一个对象属性的操作)
- var a = {age:20}
- var b = {height:30}
- Object.defineProperty(b,'age',{
- get(){
- return a.age
- },
- set(newvalue){
- a.age = newvalue
- }
- })
1、Object.defineProperty在js中的描述:
Object.defineProperty(obj, prop, descriptor) 直接在一个对象上定义一个属性,或者修改一个对象的现有 属性,并返回这个对象。
参数:obj 要在其上定义属性的对象;prop 要定义或修改的属性的名称;descriptor 将被定义或修改的属性描述符。
返回值: 传递给函数的对象obj
- html>
- <html lang="en">
- <head>
- <meta charset="UTF-8">
- <meta name="viewport" content="width=device-width, initial-scale=1.0">
- <title>数据响应式原理title>
- head>
- <body>
- <div id="app">
- Hello Vue
- div>
- <script>
- //模拟Vue实例中的data选项
- let data={
- msg:'Hello Vue'
- }
-
- //模拟Vue的实例
- let vm={};
-
- //数据劫持,当访问或者设置vm中的成员的时候,做一些干预操作
- Object.defineProperty(vm,'msg',{
- //可枚举(即可被遍历)
- enumerable:true,
- //可配置(可以使用delete删除,可以通过defineProperty重新定义)
- configurable:true,
-
- //当获取值时执行
- get(){
- console.log('getter:',data.msg);
- return data.msg;
- },
- //当设置、更新msg变量时执行
- set(newValue){
- console.log("setter:",newValue);
- if(data.msg===newValue){
- return;//前后数据相同,则不用做操作DOM的多余操作
- }
- data.msg=newValue;
- document.querySelector("#app").textContent=newValue;
- }
- })
-
- //测试setter
- vm.msg="Hello 响应式原理";
-
- //测试getter
- console.log(vm.msg);
- script>
- body>
- html>
上面实例只是对msg
这个属性实现了响应式,那Vue中data选项有多个属性,怎么做到让它们都成为响应式呢
- html>
- <html lang="en">
- <head>
- <meta charset="UTF-8">
- <meta name="viewport" content="width=device-width, initial-scale=1.0">
- <title>数据响应式原理title>
- head>
- <body>
- <div id="app">
- Hello Vue
- div>
- <script>
- //模拟Vue实例中的data选项
- let data={
- msg:'Hello Vue',
- count:0
- }
-
- //模拟Vue的实例
- let vm={};
-
- function defineProperties(data){
- //循环给每个属性使用Object.defineProperty()
- Object.keys(data).forEach(key => {
- //数据劫持,当访问或者设置vm中的成员的时候,做一些干预操作
- Object.defineProperty(vm,key,{
- //可枚举(即可被遍历)
- enumerable:true,
- //可配置(可以使用delete删除,可以通过defineProperty重新定义)
- configurable:true,
-
- //当获取值时执行
- get(){
- console.log('getter:',data[key]);
- return data[key];
- },
- //当设置、更新msg变量时执行
- set(newValue){
- console.log("setter:",newValue);
- if(data[key]===newValue){
- return;//前后数据相同,则不用做操作DOM的多余操作
- }
- data[key]=newValue;
- document.querySelector("#app").textContent=newValue;
- }
- })
- });
- }
-
- //执行该函数,使每个属性添加响应式
- defineProperties(data);
-
- //测试setter
- vm.msg="Hello 响应式原理";
-
- //测试getter
- console.log(vm.msg);
- script>
- body>
- html>
2、基于Object.defineProperty的数据劫持优势以及实现方式
Object.defineProperty的对象以及对象属性的劫持有以下优势:
(1)无需显式调用,如Vue2.x使用Object.defineProperty对象以及对象属性的劫持+发布订阅模式,只要数据发生变化直接通知变化 并驱动视图更新。
(2)可在set函数中精确得知变化数据而不用逐个遍历属性获取变化值,减少性能损耗。
实现思路:
(1)利用Object.defineProperty重新定义一遍目标对象,完成对目标对象的劫持,在属性值变化后即触发set方法 后通知订阅者,告诉该对象的某个属性值发生了变化。
(2)解析器Compile解析模板中的指令,收集指令所依赖的方法和数据,等待数据变化然后进行渲染。
(3)Watcher在收到属性值发生变化后,根据解析器Compile提供的指令进行视图渲染。
检测变化的注意事项
Object.defineProperty() 可以监测到属性的获取、修改,但是新增、删除监测不到,
- html>
- <html>
- <head>
- <meta charset="utf-8">
- <title>title>
- head>
- <body>
- <div id="app">
- <h2 @click="qq">{{isAgree.msg}}h2>
- <h2>{{isAgree.ww}}h2>
-
- <h2 v-for="(item,index) in isAgree.web" :key="index">{{item}}h2>
- <button @click="updateweb">点击button>
- div>
-
- <script src="https://cdn.jsdelivr.net/npm/vue@2.6.10/dist/vue.js">script>
- <script>
- const app = new Vue({
- el: '#app',
- data: {
- isAgree: {
- msg: 'zhangsan',
- web:['html','js']
- }
-
- },
- methods: {
- qq() {
- /* 直接添加无效
- 必须使用Vue.set或者this.$set */
- /* this.isAgree.ww = "lisi" */
- /* Vue.set(this.isAgree,'ww','lisi') */
- /* this.$set(this.isAgree,'qq','2') */
-
- /* 使用delete不可用 */
- /* console.log(this.isAgree.msg);
- delete this.isAgree.msg
- console.log(this.isAgree.msg); */
-
- /* 可以使用Vue.delete或者this.$delete
- Vue.delete(this.isAgree, 'msg') */
- /* this.$delete(this.isAgree, 'msg') */
-
-
- },
- updateweb(){
- /* this.isAgree.web[0]="vue" 无效*/
- /* this.$set(this.isAgree.web,0,'vue') 有效*/
- /* this.isAgree.web.splice(0,1,'vue') 有效*/
- }
-
- }
- })
- script>
- body>
- html>
由于 JavaScript 的限制,Vue 不能检测数组和对象的变化。尽管如此我们还是有一些办法来回避这些限制并保证它们的响应性。
Vue 无法检测 property 的添加或移除。由于 Vue 会在初始化实例时对 property 执行 getter/setter 转化,所以 property 必须在 data 对象上存在才能让 Vue 将它转换为响应式的
- var vm = new Vue({
- data:{
- a:1
- }
- })
-
- // `vm.a` 是响应式的
-
- vm.b = 2
- // `vm.b` 是非响应式的
对于已经创建的实例,Vue 不允许动态添加根级别的响应式 property。但是,可以使用 Vue.set(object, propertyName, value) 方法向嵌套对象添加响应式 property。例如,对于:
Vue.set(vm.someObject, 'b', 2)
您还可以使用 vm.$set 实例方法,这也是全局 Vue.set 方法的别名:
this.$set(this.someObject,'b',2)
有时你可能需要为已有对象赋值多个新 property,比如使用 Object.assign() 或 _.extend()。但是,这样添加到对象上的新 property 不会触发更新。在这种情况下,你应该用原对象与要混合进去的对象的 property 一起创建一个新的对象。
- // 代替 `Object.assign(this.someObject, { a: 1, b: 2 })`
- this.someObject = Object.assign({}, this.someObject, { a: 1, b: 2 })