从根源理解Proxy对比defineProperty优势,程序员面试数据结构和算法

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Web前端全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024c (备注前端)
img

正文

defineProperty和Proxy分别是Vue2和Vue3实现响应式的核心原理,实现数据响应式的关键在于对数据进行监听及操作,如以下对象obj,

let obj = {
    name: "木鱼",
    age: 18
};

当我们读取对象属性obj.name或修改对象属性obj.age=81,该如何监听到进行了obj属性的读取或修改操作的行为呢?思路是→把对象属性的读取和赋值变成一个函数,这样在函数实现读取或赋值的过程中顺便插入监听操作等逻辑代码即可实现数据截取、数据监听等功能

那么怎么把对象属性的读取赋值操作转变为一个函数呢?

Vue2实现思路:

Vue2采用了ES5提供的Object.defineProperty方法。

//接上片段代码
let obj2={}
Object.defineProperty(obj2,"name",{
    get(){
        console.log("调用了getter,则说明有人读取了name属性")
        return obj.name
    },
    set(value){
        console.log(`调用了setter,则说明有人修改了name属性为${value}`)
        obj.name=value
    }
})
console.log(obj2.name)//通过obj2拿到obj的name
obj2.name='沐萸'
console.log(obj.name)//通过修改obj2的name属性,obj的name也被同步修改

通过代理对象obj2即可对obj中的属性进行操作(即数据代理)。

那么怎么代理obj对象的所有属性呢?Vue2代码逻辑即为在一个observe函数中对obj对象进行深度遍历(利用for in对每个属性进行遍历,应用上述的Object.defineProperty逻辑代码,如果对象中有属性嵌套对象则进一步递归遍历执行)。

以上即为Vue2实现响应式数据的核心原理,Vue2通过类似上述的方式对定义的_data对象进行深度遍历,将_data对象里属性的读取赋值操作通过defineProperty都转变为一个函数(即getter、setter),并且都交给vm对象代理(vm对象即为所谓的代理对象obj2),因此对vm.name进行读取修改操作就相当于对vm._data.name进行操作,这样数据拦截、数据监听实现了,也好实现响应式数据啦~

然鹅。

Vue2实现方式的弊端:

Vue2无法监听属性的新增和删除。注意上述橙色字体尤其是加粗部分,这正是Vue2实现方式弊端的关键因素。由于它是针对每个属性进行监听,则要进行深度遍历,这会有效率的损失。而当执行了observe函数后,Vue2完成了深度遍历,_data中的数据属性都已经被改成getter、setter函数了,都被监听到了。但observe函数仅被执行了一次,这时再通过obj.newProp去新增数据对象的属性,由于observe函数已执行过,则不会再对后操作的obj新增的newProp属性进行转换为函数了,因此通过obj.newProp去新增数据对象的属性或去删除某个属性,Vue2是无法监听到该操作的。

Vue3解决Vue2弊端的关键:

Vue3同样需要把对象读取赋值转换为函数以实现监听,但Vue3不去监听对象的每个属性,而是直接监听整个对象→因此不需要进行深度遍历,只要对象变了Vue3即能监听到。

Vue3实现思路:

Vue3采用了ES6提供的Proxy方法(注意,由于该方法由ES6提供,说明Vue3也将不适用于不支持ES6以下版本的浏览器,这是Vue3的一个缺点)。

//接前面的obj对象
let handler = {
    get: function(target, prop, receiver) {
        console.log(`代理对象执行getter函数,获取目标对象的"${prop}"属性`);
        return target[prop];
    },
    set: function(target, prop, value, receiver) {
        console.log(`代理对象执行setter函数将目标对象的"${prop}"属性的值改为${value}`);
        target[prop] = value;
    }
};
let proxy = new Proxy(obj, handler);
console.log(proxy);
console.log(proxy.name); // 木鱼
console.log(proxy.age); // 18


### 总结

面试前要精心做好准备,简历上写的知识点和原理都需要准备好,项目上多想想难点和亮点,这是面试时能和别人不一样的地方。

还有就是表现出自己的谦虚好学,以及对于未来持续进阶的规划,企业招人更偏爱稳定的人。

万事开头难,但是程序员这一条路坚持几年后发展空间还是非常大的,一切重在坚持。



**前端面试题汇总**

![](https://img-blog.csdnimg.cn/img_convert/42728594459506983a38ca2b86545fc6.png)

**JavaScript**

![](https://img-blog.csdnimg.cn/img_convert/7796de226b373d068d8f5bef31e668ce.png)

**前端资料汇总**

![](https://img-blog.csdnimg.cn/img_convert/6e0ba223f65e063db5b1b4b6aa26129a.png)

**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**需要这份系统化的资料的朋友,可以添加V获取:vip1024c (备注前端)**
![img](https://img-blog.csdnimg.cn/img_convert/3d63801b8f35b9d7d1f48fd2e90e5239.png)

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**
转存中...(img-xrfmm6SB-1713452938174)]

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**
  • 22
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值