【面试题】面试官:请你说说对Vue响应式数据的理解

简介: 【面试题】面试官:请你说说对Vue响应式数据的理解

前言

我们平时的面试过程当中,问到Vue,几乎都会问到响应式的问题,因为在Vue的实现当中,响应式系统的实现就占据很大一个篇幅。这是Vue声明式编程的基石。那么如何理解响应式数据呢?相信结合源码以及手写实现会有一个更深入的理解。

大厂面试题分享 面试题库

前后端面试题库 (面试必备) 推荐:★★★★★

地址:前端面试题库  web前端面试题库 VS java后端面试题库大全

问题引出

什么是响应式数据?

响应式数据,意思是当数据发生变化后,会触发副作用的执行,从而达到某些目的。这与Vue所推崇的声明式编程简直不谋而合。利用代理模式,拦截到数据的变化,从而可以具备驱动视图变化的能力。这不就是能实现声明式编程里面核心的响应式部分了吗? 比如说,我所声明的视图里,使用到了数据,当这个数据改变后,视图跟着数据变化同步更新。

响应式数据的实现

在Vue2 当中可以这样实现

    // 定义一下基础数据对象
const obj = {
    name: "Jettsang",
    age: 25,
    arr:[1,2,3],
    msg:{
        phoneCode:'12345678901',
        address:'唐宁街10号'
    }
}
// vue 2 定义响应式,其实对defineProperty的封装
function defineReactive(target,key ,value){
    // 对value进行递归定义响应式(如果value是对象的话)
    observer(value)
    Object.defineProperty(target,key,{
        get(){
            // 在这里收集依赖,记录下watcher
            return value
        },
        set(newVal){
            if(value !== newVal){
                value = newVal
                // 别忘了对新的值定义响应式,因为新的值可能是对象
                observer(newVal)
                // 触发watcher通知视图更新
            }
        }
    })
}
function observer(data){
    if(typeof data !=='object' || typeof data == null){
        return data
    }
// 这里是enumerable对象,用for in 遍历
    for(let key in data )
    {
    // 定义响应式
        defineReactive(data,key,data[key])
    }
}
observer(obj)
复制代码

可以在控制台看到

其实对数组也是可以进行defineProperty的,但Vue的设计者明显考虑了性能和使用上的平衡,因此对数组做了特殊处理,待会在源码揭秘环节可以更细节的看到。

补充完善对数组的响应式处理 vue2当中,对会改变数组本身的方法(shift unshift push pop splice sort reverse)进行了重写,从而实现数组的响应式

function observer(data) {
    if (typeof data !== 'object' || typeof data == null) {
        return data
    }
    // 增加对数组的特殊处理
    if (Array.isArray(data)) {
        setArrayProto(data)
        return
    }
    for (let key in data) {
        defineReactive(data, key, data[key])
    }
}
// 定义需要重写的方法
const OVERRIDE_METHODS = ['push', 'pop', 'unshift', 'shift', 'reverse', 'sort', 'splice']
// 定义重写proto上数组方法的函数
function setArrayProto(data) {
    const prototype = Array.prototype
    // 这里的create方法会将 newPrototype的原型指向prototype
    // 因此通过原型链接仍然可以获取其他的Array原型方法
    const newPrototype = Object.create(prototype)
    OVERRIDE_METHODS.forEach(method=>{
        newPrototype[method] = function(...args){
            // 这里可以加入响应式逻辑
            // 。。。
            // 注意this的指向
            prototype[method].call(this,...args)
        }
    })
    data.__proto__ = newPrototype
}
复制代码

可以看到 ,新的原型已经挂在上面了

Vue 2 当中的缺陷是什么?

不难看出 ,Vue2当中的缺陷基本上是由于defineProperty的局限性导致的,总结一下是:

  1. defineProperty需要递归去添加getter和setter,比较浪费性能
  2. 权衡之后数组不采用defineProperty,而是重写了原型上的会改变数组本身的方法(shift unshift push pop splice sort reverse)
  3. 新增和删除不能实现被监听到,需要额外的���和set和delete 这两个API
  4. 对ES6的新数据结构Map和Set不支持响应式

那么Vue 3当中又做了哪些改变呢?

Vue 3 基于 ProxyAPI可以去除以上缺点

Vue3 采用Proxy这个API来实现数据的代理,从而实现整个响应式系统,抛弃了一些兼容性,同时获得更好的性能提升

function reactive(target){
    return new Proxy(target,{
        get(target,key){
            const value = target[key]
            // 如果是对象,需要代理该对象
            // 注意了,这里是触发getter才会去响应式,可以节省性能
            if(typeof value =='object'){
                return reactive(value)
            }
            // 这里可以加收集依赖的逻辑了
            // 。。。
            return Reflect.get(target,key)
        },
        set(target,key,value){
            const oldValue = target[key]
            // 如果值不相等,才去设置
            if(oldValue !== value){
                Reflect.set(target,key,value)
            }
            // 触发副作用函数:比如刷新视图
            // 。。。
            return true
        }
    })
}
const proxyObj = reactive(obj)
console.log(proxyObj);
复制代码

看控制台 ,打印出来的msg里并不是响应式,只有当你去触发getter的时候,才会返回响应式代理对象,可以节省性能,是懒散的代理。

源码探秘

Vue2

这边只看核心逻辑,对边界情况以及一些细节的处理可以忽略,看官们有空自己研究哈

源码位于 src/core/observer/index.ts

定义的这个Observer当中,去对数据做一个观测

defineReactive函数

src/core/observer/array.ts 对数组的处理

Vue3

位于packages/reactivity/src/reactive.ts

createReactiveObject函数

collectionHandlers 是处理map set这种集合的处理器/ 和baseHandlers 基础处理器

看看base处理器里面定义的getter

可以看到懒代理的实现

总结

Vue2到3的响应式数据的处理是基于其核心API来进行具体的优化和权衡取舍。 从2的defineproperty通过递归劫持property,到权衡性能对数组做特殊处理,到3采用proxy同时使用懒代理来优化性能,其中无不体现出Vue设计者的精彩构思。读源码就像读一本书,每次都都会有新的发现。

大厂面试题分享 面试题库

前后端面试题库 (面试必备) 推荐:★★★★★

地址:前端面试题库  web前端面试题库 VS java后端面试题库大全

相关文章
|
2月前
|
缓存 NoSQL 关系型数据库
美团面试:MySQL有1000w数据,redis只存20w的数据,如何做 缓存 设计?
美团面试:MySQL有1000w数据,redis只存20w的数据,如何做 缓存 设计?
美团面试:MySQL有1000w数据,redis只存20w的数据,如何做 缓存 设计?
|
4月前
|
SQL JavaScript 前端开发
Vue实现动态数据透视表(交叉表)
Vue实现动态数据透视表(交叉表)
256 13
|
4月前
|
JavaScript 前端开发 UED
vue2和vue3的响应式原理有何不同?
大家好,我是V哥。本文详细对比了Vue 2与Vue 3的响应式原理:Vue 2基于`Object.defineProperty()`,适合小型项目但存在性能瓶颈;Vue 3采用`Proxy`,大幅优化初始化、更新性能及内存占用,更高效稳定。此外,我建议前端开发者关注鸿蒙趋势,2025年将是国产化替代关键期,推荐《鸿蒙 HarmonyOS 开发之路》卷1助你入行。老项目用Vue 2?不妨升级到Vue 3,提升用户体验!关注V哥爱编程,全栈开发轻松上手。
231 2
|
4月前
|
JavaScript 前端开发 API
管理数据必备;侦听器watch用法详解,vue2与vue3中watch的变化与差异
一篇文章同时搞定Vue2和Vue3的侦听器,是不是很棒?不要忘了Vue3中多了一个可选项watchEffect噢。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
4月前
|
存储 数据采集 供应链
属性描述符初探——Vue实现数据劫持的基础
属性描述符还有很多内容可以挖掘,比如defineProperty与Proxy的区别,比如vue2与vue3实现数据劫持的方式有什么不同,实现效果有哪些差异等,这篇博文只是入门,以后有时间再深挖。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
5月前
|
缓存 安全 Java
面试中的难题:线程异步执行后如何共享数据?
本文通过一个面试故事,详细讲解了Java中线程内部开启异步操作后如何安全地共享数据。介绍了异步操作的基本概念及常见实现方式(如CompletableFuture、ExecutorService),并重点探讨了volatile关键字、CountDownLatch和CompletableFuture等工具在线程间数据共享中的应用,帮助读者理解线程安全和内存可见性问题。通过这些方法,可以有效解决多线程环境下的数据共享挑战,提升编程效率和代码健壮性。
176 6
|
8月前
|
缓存 JavaScript UED
Vue3中v-model在处理自定义组件双向数据绑定时有哪些注意事项?
在使用`v-model`处理自定义组件双向数据绑定时,要仔细考虑各种因素,确保数据的准确传递和更新,同时提供良好的用户体验和代码可维护性。通过合理的设计和注意事项的遵循,能够更好地发挥`v-model`的优势,实现高效的双向数据绑定效果。
268 64
|
6月前
|
NoSQL 关系型数据库 MySQL
招行面试:高并发写,为什么不推荐关系数据?
资深架构师尼恩针对高并发场景下为何不推荐使用关系数据库进行数据写入进行了深入剖析。文章详细解释了关系数据库(如MySQL)在高并发写入时的性能瓶颈,包括存储机制和事务特性带来的开销,并对比了NoSQL数据库的优势。通过具体案例和理论分析,尼恩为读者提供了系统化的解答,帮助面试者更好地应对类似问题,提升技术实力。此外,尼恩还分享了多个高并发系统的解决方案及优化技巧,助力开发者在面试中脱颖而出。 文章链接:[原文链接](https://mp.weixin.qq.com/s/PKsa-7eZqXDg3tpgJKCAAw) 更多技术资料和面试宝典可关注【技术自由圈】获取。
|
7月前
|
存储 Java easyexcel
招行面试:100万级别数据的Excel,如何秒级导入到数据库?
本文由40岁老架构师尼恩撰写,分享了应对招商银行Java后端面试绝命12题的经验。文章详细介绍了如何通过系统化准备,在面试中展示强大的技术实力。针对百万级数据的Excel导入难题,尼恩推荐使用阿里巴巴开源的EasyExcel框架,并结合高性能分片读取、Disruptor队列缓冲和高并发批量写入的架构方案,实现高效的数据处理。此外,文章还提供了完整的代码示例和配置说明,帮助读者快速掌握相关技能。建议读者参考《尼恩Java面试宝典PDF》进行系统化刷题,提升面试竞争力。关注公众号【技术自由圈】可获取更多技术资源和指导。
|
8月前
|
JavaScript 前端开发 开发者
Vue是如何劫持响应式对象的
Vue是如何劫持响应式对象的
129 18