vue3响应式原理

  • 回顾vue2Object.defineProperty

  • 缺点

    • 深度监听对象需要一次性递归
    • 无法监听新增属性、删除属性(Vue.setVue.delete)
    • 无法监听原生数组,需要特殊处理
  • 学习proxy语法

  • Vue3中如何使用proxy实现响应式

Proxy 基本使用

js 复制代码
// const data = {
//     name: 'zhangsan',
//     age: 20,
// }
const data = ['a', 'b', 'c']

const proxyData = new Proxy(data, {
    get(target, key, receiver) {
        // 只处理本身(非原型的)属性
        const ownKeys = Reflect.ownKeys(target)
        if (ownKeys.includes(key)) {
            console.log('get', key) // 监听
        }

        const result = Reflect.get(target, key, receiver)
        return result // 返回结果
    },
    set(target, key, val, receiver) {
        // 重复的数据,不处理
        if (val === target[key]) {
            return true
        }

        const result = Reflect.set(target, key, val, receiver)
        console.log('set', key, val)
        // console.log('result', result) // true
        return result // 是否设置成功
    },
    deleteProperty(target, key) {
        const result = Reflect.deleteProperty(target, key)
        console.log('delete property', key)
        // console.log('result', result) // true
        return result // 是否删除成功
    }
})

vue3响应式原理

深度监听,性能更好(获取到哪一层才触发响应式get,不是一次性递归)

可监听新增/删除属性

可监听数组变化

javascript 复制代码
// 创建响应式
function reactive(target = {}) {
    if (typeof target !== 'object' || target == null) {
        // 不是对象或数组,则返回
        return target
    }

    // 代理配置
    const proxyConf = {
        get(target, key, receiver) {
            // 只处理本身(非原型的)属性
            const ownKeys = Reflect.ownKeys(target)
            if (ownKeys.includes(key)) {
                console.log('get', key) // 监听
            }
    
            const result = Reflect.get(target, key, receiver)
        
            // 深度监听
            // 性能如何提升的?获取到哪一层才触发响应式get,不是一次性递归
            return reactive(result)
        },
        set(target, key, val, receiver) {
            // 重复的数据,不处理
            if (val === target[key]) {
                return true
            }
    
            const ownKeys = Reflect.ownKeys(target)
            if (ownKeys.includes(key)) {
                console.log('已有的 key', key)
            } else {
                console.log('新增的 key', key)
            }

            const result = Reflect.set(target, key, val, receiver)
            console.log('set', key, val)
            // console.log('result', result) // true
            return result // 是否设置成功
        },
        deleteProperty(target, key) {
            const result = Reflect.deleteProperty(target, key)
            console.log('delete property', key)
            // console.log('result', result) // true
            return result // 是否删除成功
        }
    }

    // 生成代理对象
    const observed = new Proxy(target, proxyConf)
    return observed
}

// 测试数据
const data = {
    name: 'zhangsan',
    age: 20,
    info: {
        city: 'shenshen',
        a: {
            b: {
                c: {
                    d: {
                        e: 100
                    }
                }
            }
        }
    }
}
const proxyData = reactive(data)
相关推荐
dasseinzumtode1 分钟前
在 React 项目中使用 TypeScript 编写 Service Worker 的完整方案
前端·react.js
来一首_We_go2 分钟前
React Router V7 踩坑指南
前端·react.js
出师未捷的小白3 分钟前
我的嘛也!实现一个时间轴无缝轮播居然如此简单......
前端
郡杰5 分钟前
JavaScript基础语法
前端·javascript
cypking17 分钟前
svelte+vite+ts+melt-ui从0到1完整框架搭建
前端
WebInfra25 分钟前
Rspack 加入 Next.js 生态:携手改进前端工具链
前端·javascript·github
三小河25 分钟前
Web Speech API实现语音的识别和播放
前端
华科云商xiao徐26 分钟前
使用libcurl编写爬虫程序指南
前端
伶俜monster26 分钟前
破解旋转死锁:Threejs 四元数魔法对抗欧拉角困局
前端·webgl·three.js
ONE_Gua27 分钟前
魔改chromium源码——canvas指纹修改 第一节
前端·chrome·浏览器