手撸vue3核心源码——响应式原理(ProxyRefs)

简介: 手撸vue3核心源码——响应式原理(ProxyRefs)

ProxyRefs初步实现

先来看一下单测要实现的功能, 这里我们先实现一下get功能, 我们可以看到这里的user对象里面包裹的有一个ref对象,我们这里取name时要实现不用.value,就可以拿到ref对象的值


it("it test ProxyRefs", () => {
        const user = {
            name: ref("meng"),
            age: 18
        }
        const ProxyUser = ProxyRefs(user)
        expect(user.name.value).toBe("meng")
        expect(ProxyUser.name).toBe("meng")
        expect(ProxyUser.age).toBe(18)
    })

顺着思路来想

我们一个对象里面只要有value为ref类型的key ,那么我们取的时候就不用.value,我们想想取的时候是不是要触发get操作,我们这里的proxyRefs也会将user转化成Proxy对象,那么我们在handler的get操作即可


export function ProxyRefs(objectWithRef) {
    return new Proxy(objectWithRef, {
        get(target, key) {
            const res = unRef(Reflect.get(target, key))
            return res
        }
    })
}

我们上一章实现了unRef也就是直接ref类型数据的值,这里刚好我们取出来值就是一个ref类型数据,然后给它做unRef处理就可以得到了


ProxyRefs完善

上面我们只做了get操作,接下来我们实现一下set操作,先来看一下单测,当我们修改了ProxyUser里面属性的值时,也会改变user的属性的值,并且对于改变的值若还是ref类型我们应该替换,如果不是了,那么我们应该对.value重新赋值了


it("it test ProxyRefs", () => {
        const user = {
            name: ref("meng"),
            age: 18
        }
        const ProxyUser = ProxyRefs(user)
        expect(user.name.value).toBe("meng")
        expect(ProxyUser.name).toBe("meng")
        expect(ProxyUser.age).toBe(18)
        ProxyUser.name = ref("chao")
        expect(ProxyUser.name).toBe("chao")
        expect(user.name.value).toBe("chao")
        ProxyUser.name = "40"
        expect(ProxyUser.name).toBe("40")
        expect(user.name.value).toBe("40")
    })

顺着思路写一下set操作,我们分了两种情况来写,我们修改后的值是一个ref还是普通的,对于ref我们进行覆盖也就是用Reflect.set方法,如果是普通的我们就对.value重新赋值


export function ProxyRefs(objectWithRef) {
    return new Proxy(objectWithRef, {
        get(target, key) {
            const res = unRef(Reflect.get(target, key))
            return res
        },
        set(target, key, value) {
            if (isRef(target[key]) && !isRef(value)) {
                return target[key].value = value
            } else {
                return Reflect.set(target, key, value)
            }
        }
    })
}


通过这样我们就可以得到一个PrxoxyRefs了,涉及到Proxy的其实操作都是一样的套路,分情况直接操作即可。


相关文章
|
2月前
|
缓存 JavaScript UED
Vue3中v-model在处理自定义组件双向数据绑定时有哪些注意事项?
在使用`v-model`处理自定义组件双向数据绑定时,要仔细考虑各种因素,确保数据的准确传递和更新,同时提供良好的用户体验和代码可维护性。通过合理的设计和注意事项的遵循,能够更好地发挥`v-model`的优势,实现高效的双向数据绑定效果。
154 64
|
2月前
|
JavaScript 前端开发 API
Vue 3 中 v-model 与 Vue 2 中 v-model 的区别是什么?
总的来说,Vue 3 中的 `v-model` 在灵活性、与组合式 API 的结合、对自定义组件的支持等方面都有了明显的提升和改进,使其更适应现代前端开发的需求和趋势。但需要注意的是,在迁移过程中可能需要对一些代码进行调整和适配。
122 60
|
16天前
|
JavaScript API 数据处理
vue3使用pinia中的actions,需要调用接口的话
通过上述步骤,您可以在Vue 3中使用Pinia和actions来管理状态并调用API接口。Pinia的简洁设计使得状态管理和异步操作更加直观和易于维护。无论是安装配置、创建Store还是在组件中使用Store,都能轻松实现高效的状态管理和数据处理。
60 3
|
2月前
|
前端开发 JavaScript 测试技术
Vue3中v-model在处理自定义组件双向数据绑定时,如何避免循环引用?
Web 组件化是一种有效的开发方法,可以提高项目的质量、效率和可维护性。在实际项目中,要结合项目的具体情况,合理应用 Web 组件化的理念和技术,实现项目的成功实施和交付。通过不断地探索和实践,将 Web 组件化的优势充分发挥出来,为前端开发领域的发展做出贡献。
45 8
|
2月前
|
存储 JavaScript 数据管理
除了provide/inject,Vue3中还有哪些方式可以避免v-model的循环引用?
需要注意的是,在实际开发中,应根据具体的项目需求和组件结构来选择合适的方式来避免`v-model`的循环引用。同时,要综合考虑代码的可读性、可维护性和性能等因素,以确保系统的稳定和高效运行。
41 1
|
2月前
|
JavaScript
Vue3中使用provide/inject来避免v-model的循环引用
`provide`和`inject`是 Vue 3 中非常有用的特性,在处理一些复杂的组件间通信问题时,可以提供一种灵活的解决方案。通过合理使用它们,可以帮助我们更好地避免`v-model`的循环引用问题,提高代码的质量和可维护性。
49 1
|
2月前
|
JavaScript
在 Vue 3 中,如何使用 v-model 来处理自定义组件的双向数据绑定?
需要注意的是,在实际开发中,根据具体的业务需求和组件设计,可能需要对上述步骤进行适当的调整和优化,以确保双向数据绑定的正确性和稳定性。同时,深入理解 Vue 3 的响应式机制和组件通信原理,将有助于更好地运用 `v-model` 实现自定义组件的双向数据绑定。
|
2月前
|
JavaScript 前端开发 API
从Vue 2到Vue 3的演进
从Vue 2到Vue 3的演进
52 0
|
2月前
|
JavaScript 前端开发 API
Vue.js响应式原理深度解析:从Vue 2到Vue 3的演进
Vue.js响应式原理深度解析:从Vue 2到Vue 3的演进
71 0
|
8月前
|
JavaScript API
【vue实战项目】通用管理系统:api封装、404页
【vue实战项目】通用管理系统:api封装、404页
85 3
下一篇
开通oss服务