经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JS/JS库/框架 » Vue.js » 查看文章
vue3响应式Proxy与Reflect的理解及基本使用实例详解
来源:jb51  时间:2022/8/2 12:45:36  对本文有异议

正文

在第四章中,作者讲述了Vue.js中响应式系统的设计与实现,这一块其实是整个框架的基石,也是MVVM中,ViewModel(VM)的重要组成部分。 其实在上一章中我已经感觉很难了,有一些操作作者也只是几笔带过,却很值得我们思考。这一张中,我们将目光着重于响应式数据本身,来完善上一章中我们的demo。

理解Proxy与Reflect

vue3的响应式离不开Proxy,说到Proxy则离不开Reflect.这两个对象是ES6新增的对象,同时在编程领域,他们也代表着2种设计模式,即代理与反射。设计模式都是后话,等我学会了也一定写一篇文章分享一下??.

Proxy

Proxy 可以理解成,在目标对象之前架设一层“拦截”,外界对该对象的访问,都必须经过这层拦截,而我们就可以通过这层拦截去改变目标对象的内容或者行为,或者叫过滤和控制。这个词本意就是代理,好比一个代理人站在神奇,我们所有行为都会被他过滤,可能我们说的话,经过代理人一说,意思就变了。

ES6 原生提供 Proxy 构造函数,用来生成 Proxy 实例。

  1. var proxy = new Proxy(target, handler);

其中target表示要代理的那个对象,handler则是表示我们需要拦截的行为,这里直接放一张阮一峰的截图。

Reflect

Reflect中文译为:反射。如果说Proxy 是有一个代理人站在身前面,帮你拦截并处理一些行为,那么Reflect就是你身后的一面镜子,它能看见真实的自己。

而你自己,就是一个类或者对象,或者一个函数,只要是js中存在的,都能被ProxyReflect处理。

它的操作和Proxy正好相反,但却一一对应。比如我们获取对象中一个属性。

  1. const obj = {foo:1}
  2. const a = Reflect.get(obj, 'foo')

这一小节主要是介绍了Proxy与Reflect,后面会有一个应用老告诉你为什么Proxy与Reflect与响应式数据息息相关。

实践示例

看完了ProxyReflect的基本使用之后,我们实践一下。

我们曾经写过这样的代码

  1. const reactive = (object)=>{
  2. return new Proxy(object,{
  3. get(target,key){
  4. track(target,key)
  5. return target[key]
  6. }
  7. set(target,key, newVal){
  8. target[key] = newVal
  9. trigger(target,key)
  10. return true
  11. }
  12. })
  13. }

其实就是用Proxy代理了对象读和取操作,在读的时候收集依赖,在取的时候触发响应。看起来似乎没有问题,那么我们再试继续往下写

  1. const obj = {
  2. a:1,
  3. get b(){
  4. return this.a
  5. }
  6. }
  7. const data = reactive(obj)
  8. effect(()=>{
  9. console.log(data.b)
  10. })
  11. setTimeOut(()=>{
  12. data.b++
  13. },500)

这里我们没有用一般的对象写法,而是通过访问器为它新增了一个b属性.之后,我们先把这个对象转换为响应式对象,再给他们设定一个响应式的回调,然后在冬天改变他的值,理论上这时候应该会执行副作用函数,但是实际上呢,根本不会执行。

我们回顾一下之前写的reactive方法,在里面返回的是target[key],当我们的target是obj,key是b的时候,那个this会是谁呢?因为target是原始对象,也就是obj,根据谁调用是谁的原则,这个this也就指向了obj。obj是响应式对象吗,显然不是,那个b也就永远不会执行副作用函数,响应式就失效了。

这里其实就是this的指向问题,你可能会说一般人怎么会用getter去赋值属性呢,但是这个作为一个简单的case,甚至都算不上边界,我们需要解决它。

解决的方法也很简单,就是通过Reflect。这也是为什么我说ProxyReflect就是焦不离孟 孟不离焦. 我们的reactive,get的时候,加入第三个参数receiver

  1. get(target,key){
  2. track(target,key,receiver)
  3. return Reflect.get(target,key,receiver)
  4. }

我这里理解的是,receiver就相当于函数的bind方法,它改变的this的执行,当我们同过Reflect读取值的时候,this的指向被改为receiver,而Reflect时的receiver又是Proxy中的入参,它执行了这个Proxy,从而把前文中this的指向由obj改为data,这样响应式就不会丢失了。

下一章中,作者会深入的介绍一下Proxy和一些相关的标准。

以上就是vue3响应式Proxy与Reflect的理解及基本使用实例详解的详细内容,更多关于vue3响应式Proxy Reflect的资料请关注w3xue其它相关文章!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号