响应式框架原理3(Object.defineProperty V

2020-03-04  本文已影响0人  nomooo

响应式框架原理2(数据劫持与代理-监听数组变化)
继续之前的记录、、
使用 Proxy 来完成代码重构:

            let data = {
                stage: '状态',
                course: {
                    title: '标题',
                    author: ['作者1', '作者2'],
                    publishTime: '出版时间'
                }
            }

            const observe = data => {
                if (!data || Object.prototype.toString.call(data) !== '[object Object]') {
                    return
                }

                Object.keys(data).forEach(key => {
                    let currentValue = data[key]
                    // proxy 也可以对函数类型进行代理。这里只对承载数据类型的 object 进行处理
                    if (typeof currentValue === 'object') {
                        observe(currentValue)
                        data[key] = new Proxy(currentValue, {
                            set(target, property, value, receiver) {
                                // 因为数组的 push 会引起 length 属性的变化,所以 push 之后会触发两次 set 操作,只需要保留一次即可,property 为 length 时,忽略
                                if (property !== 'length') {
                                    console.log(`setting ${key} value now, setting value is`, currentValue)
                                }
                                return Reflect.set(target, property, value, receiver)
                            }
                        })
                    } else {
                        Object.defineProperty(data, key, {
                            enumerable: true,
                            configurable: false,
                            get() {
                                console.log(`getting ${key} value now, getting value is:`, currentValue)
                                return currentValue
                            },
                            set(newValue) {
                                currentValue = newValue
                                console.log(`setting ${key} value now, setting value is`, currentValue)
                            }
                        })
                    }
                })
            }
            observe(data)

对数组操作:

            data.course.author.push('作者3')

上面代码在使用proxy进行代理时,并没有对getter进行代理,所以输出结果不会有getting value输出
使用 Object.defineProperty;对于键值为对象类型的情况,继续递归调用 observe 方法,并通过 Proxy 返回的新对象对 data[key] 重新赋值,这个新值的 getter 和 setter 已经被添加了代理。

结合上两篇笔记,对 Proxy 实现数据代理和 Object.defineProperty 实现数据拦截进行对比:

上一篇 下一篇

猜你喜欢

热点阅读