Vue2.X监听data变化的核心API—Object.defineProperty详解

Vue2.X监听data变化的核心API—Object.defineProperty基本使用:

Object.defineProperty实现响应式

1.监听对象(简单对象)

 

 上面通过监听get,set方法了解到data变化,进而可以达到响应式。

2.复杂对象(深度监听),深度监听

触发更新视图
// 触发更新视图
function updateView() {
    console.log('视图更新')
}

在上面例子data加:

// 准备数据
const data = {
    name: '佩奇',
    age: 20,
    info: {
        address: '宁波'  // 需要深度监听
    },
}
// 重新定义属性,监听起来
function defineReactive(target, key, value) {
    // 核心 API
    Object.defineProperty(target, key, {
        get() {
            return value
        },
        set(newValue) {
            if (newValue !== value) {
                // 设置新值
                // 注意,value 一直在闭包中,此处设置完之后,再 get 时也是会获取最新的值
                value = newValue

                // 触发更新视图
                updateView()
            }
        }
    })
}

// 监听对象属性
function observer(target) {
    if (typeof target !== 'object' || target === null) {
        // 不是对象或数组
        return target
    }
    // 重新定义各个属性(for in 也可以遍历数组)
    for (let key in target) {
        defineReactive(target, key, target[key])
    }
}
// 监听数据
observer(data)
 

此时我们在上面例子代码,没有监听到。



 

此时优化一下,在defineReactive方法里加一层监听。

// 重新定义属性,监听起来
function defineReactive(target, key, value) {
    // 深度监听
    observer(value)
    // 核心 API
    Object.defineProperty(target, key, {
        get() {
            return value
        },
        set(newValue) {
            if (newValue !== value) {
              
                // 设置新值
                // 注意,value 一直在闭包中,此处设置完之后,再 get 时也是会获取最新的值
                value = newValue

                // 触发更新视图
                updateView()
            }
        }
    })
}

      可以深度监听到了。

 

       上面二例子, data.age = {num:21}  可以监听到  而data.age.num = 22 却没有监听到。

      此时优化一下,在defineReactive方法中Object.defineProperty里set方法加 深度监听  observer(newValue)加一层监听。

// 重新定义属性,监听起来
function defineReactive(target, key, value) {
    // 深度监听
    observer(value)

    // 核心 API
    Object.defineProperty(target, key, {
        get() {
            return value
        },
        set(newValue) {
            if (newValue !== value) {
                 // 深度监听
                 observer(newValue)

                // 设置新值
                // 注意,value 一直在闭包中,此处设置完之后,再 get 时也是会获取最新的值
                value = newValue

                // 触发更新视图
                updateView()
            }
        }
    })
}

  监听数组:

3.几个缺点

1.深度监听,需要递归到底,一次性计算量大(通过上面的例子)
2.
新增属性,监听不到 —— 所以有 Vue.set

3.删除属性,监听不到 —— 所有已 Vue.delete

原文地址:https://www.cnblogs.com/aixue/p/12687115.html