zoukankan      html  css  js  c++  java
  • 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

  • 相关阅读:
    并发与并行
    OpenCV 图像集合操作
    C++ 输出时间
    绘制模型图
    检测图像文件是否损坏
    QImage,Mat ,QByteArray转换
    图像拼接3
    图像拼接2】
    图像拼接 Stitcher
    《将博客搬至CSDN》
  • 原文地址:https://www.cnblogs.com/aixue/p/12687115.html
Copyright © 2011-2022 走看看