用一段代码明白vue3响应式

vue的响应式的逻辑简单的说明:对象get时进行添加副作用函数,set时执行副作用函数

当我们对obj.text进行改变时需要页面刷新,我们需要重新执行这句代码,那么这就是副作用函数

javascript 复制代码
// 副作用函数
function effect(){
	document.body.textContent = obj.text
}

副作用函数执行,创建proxy触发get收集副作用函数,set时执行副作用函数,

javascript 复制代码
let bucket = []
function reactive (target) {
    return new Proxy(target, {
        get (target, key, receiver) {
            bucket.push(effect)
            return target[key]
        },
        set (target, key, newVal, receiver) {
            target[key] = newVal
            bucket.forEach(effect =>{
                effect()
            })
            return true
        }
    })
}

但是副作用函数需要变为多更为灵活,所以改写effect,定义全局变量activeEffect用于记录副作用函数

javascript 复制代码
let activeEffect 
let bucket = []
function effect(fn){
    activeEffect = fn
    // 思考为什么要执行一次
    fn()
}
function reactive (target) {
    return new Proxy(target, {
        get (target, key, receiver) {
            bucket.push(activeEffect)
            return target[key]
        },
        set (target, key, newVal, receiver) {
            target[key] = newVal
            bucket.forEach(effect =>{
                effect()
            })
            return true
        }
    })
}

但是使用会发现一个问题,就是bucket会重复收集副作用函数,因此我们需要使用Set()来避免重复记录副作用函数,而且如果同时存在两个key都有副作用函数,我们是无法分辨的

javascript 复制代码
function effect1(){
	document.getElementById('text').textContent = _obj.text
}
function effect2(){
	document.getElementById('text2').textContent = _obj.text2
}

此时我们的副作用函数是要与key做对应

css 复制代码
_obj
   |
   ----text---effect1
   |
   ----text2---effect2

改写如下:

javascript 复制代码
let activeEffect 
let bucket = new WeakMap()
function effect(fn){
    activeEffect = fn
    fn()
}
function reactive (target) {
    return new Proxy(target, {
        get (target, key, receiver) {
            if (!bucket.get(target)) {
                bucket.set(target, new Map())
            }
            if (!bucket.get(target).get(key)) {
                bucket.get(target).set(key, new Set())
            }
            bucket.get(target).get(key).add(activeEffect)
            return target[key]
        },
        set (target, key, newVal, receiver) {
            target[key] = newVal
            if (!bucket.get(target)) return
            if (!bucket.get(target).get(key)) return
            bucket.get(target).get(key).forEach(effect => {
                effect()
            })
            return true
        }
    })
}

WeakMap可以搜一下它的解释,简单来说就是它可以对不用的key触发垃圾回收机制,Map则不会

我们提取get里面的处理逻辑到track函数中,set里面的处理逻辑到trigger中

javascript 复制代码
function effect (fn) {
    activeEffect = fn
    fn()
}
function tarck (target, key) {
    if (!bucket.get(target)) {
        bucket.set(target, new Map())
    }
    if (!bucket.get(target).get(key)) {
        bucket.get(target).set(key, new Set())
    }
    bucket.get(target).get(key).add(activeEffect)
}
function trigger (target, key) {
    if (!bucket.get(target)) return
    if (!bucket.get(target).get(key)) return
    bucket.get(target).get(key).forEach(effect => {
        effect()
    })
}
function reactive (target) {
    return new Proxy(target, {
        get (target, key, receiver) {
            tarck(target, key, receiver)
            return target[key]
        },
        set (target, key, newVal, receiver) {
            target[key] = newVal
            trigger(target, key, newVal, receiver)
            return true
        }
    })
}

使用:

javascript 复制代码
let _obj = reactive({
    text: '123',
    text2: '123'
})
effect(() => {
    document.getElementById('text').textContent = _obj.text
    document.getElementById('text2').textContent = _obj.text2
})
相关推荐
小彭努力中4 小时前
199.Vue3 + OpenLayers 实现:点击 / 拖动地图播放音频
前端·vue.js·音视频·openlayers·animate
慧一居士5 小时前
Vue项目中,何时使用布局、子组件嵌套、插槽 对应的使用场景,和完整的使用示例
前端·vue.js
Jave21086 小时前
实现全局自定义loading指令
前端·vue.js
慧一居士11 小时前
Vue项目中,子组件调用父组件方法示例,以及如何传值示例,对比使用插槽和不使用插槽区别
前端·vue.js
SeSs IZED11 小时前
【Nginx 】Nginx 部署前端 vue 项目
前端·vue.js·nginx
SuperEugene12 小时前
Vue3 性能优化规范:日常必做优化(不玄学、可落地)|可维护性与兜底规范篇
开发语言·前端·javascript·vue.js·性能优化·前端框架
A923A12 小时前
【小兔鲜电商前台 | 项目笔记】第二天
前端·vue.js·笔记·项目·小兔鲜
happymaker062613 小时前
vue指令扩展以及监视器的使用
前端·javascript·vue.js
还是大剑师兰特13 小时前
EventBus核心方法用法
javascript·vue.js·大剑师
happymaker062615 小时前
vue中对list的函数处理方式总结,以及常见功能的实现方法
javascript·vue.js·list