js对象属性的拦截与Proxy代理与Reflect映射的用法和区别讲解

 更新时间:2023年06月01日 10:09:34   作者:坚毅的小解同志  
reflect和proxy都是JavaScript中用于处理对象的特殊API,下面这篇文章主要给大家介绍了关于js对象属性的拦截与Proxy代理与Reflect映射的用法和区别,文中通过代码介绍的非常详细,需要的朋友可以参考下

对象属性的拦截

介绍

在对象中,set 和 get 是属性的特性,用于定义属性的赋值和取值行为。它们允许您在属性被赋值或取值时执行自定义的逻辑。

Set

set 是一个对象属性的特性,用于定义属性的赋值行为。当给属性赋值时,set 方法会被调用,允许执行自定义的逻辑。

  let obj = {
            name:0,
            set changename(value){
                if (value >5) {
                     this.name =value;
                }
            }
        }
obj.changename=10  //10
obj.changename=4    //0
console.log(obj.name);

通过changename函数 在里面进行需要的条件判断来修改name值,给这个函数赋值的时候,就会触发这个set。

注意 函数名跟属性名不能一样,不然就要递归调用报栈溢出错误了

Get

get 用于定义对象属性的获取行为。当访问对象的属性时,get 方法会被触发,并且可以在方法中执行相应的逻辑。

  let obj = {
            name:0,
            set changename(value){
                if (value >5) {
                   this.name =value;
                }
            },
            get changename(){
               return this.name =1
            },
        }
obj.changename=10  //10
obj.changename=4    //0
console.log(obj.changename); //1  (获取的时候会触发get)

假设 我们要访问一个属性,然后不管设置还有获取都需要对这个属性进行相应操作 ,我们就可以使用 很好的搭配使用set和get。 其实就相当于你针对一个属性写了两个不同的函数进行操作,只不过 set get 我们写的函数名称一样,便于易读浏览。

对象的拦截

介绍

Proxy 是 JavaScript 提供的一个内置对象,用于创建一个代理对象,可以拦截并自定义对目标对象的操作。通过使用 Proxy,我们可以对目标对象的属性访问、赋值、删除等操作进行拦截和处理。

使用

   let obj = {
        name: 0,
      };
      let handle = {
        set: function (target, name, value) {
          if (value > 5) {
            target[name] = value;
          }
        },
        get: function (target, name) {
          return (target[name] = 1);
        },
      };
      let proxy = new Proxy(obj, handle);
      proxy.changename = 10; //10
      proxy.changename = 4; //0
      console.log(proxy.changename); //1

通过proxy代理来访问 对象进行获取和赋值等操作。

对象属性拦截和对象拦截区别

区别就跟名字一样,对象属性拦截,是你在对象中给某个属性设置拦截操作,当它获取,和赋值的时候触发,对象拦截 是设置proxy代理,通过代理来访问,既然通过代理来访问对象属性,则任何操作,只要你对这个对象有操作,都会进行拦截判断,
也就是说 一个是针对对象里的一个属性,一个是针对整个对象。

练习题

   let obj = {
          name: 0,
          set changename(value) {
            if (value > 5) {
              this.name = value;
            }
          },
          get changename() {
            return (this.name += 1);
          },
        };
        let handle = {
          //target原对象,name是属性
          get: function (target, name) {
            target[name] += 1;
            //注意 get 一定要return
            return target[name];
          },
          //value传的值
          set: function (target, name, value) {
            if (typeof value == "number") {
              target[name] = value;
            }
          },
        };
        let proxy = new Proxy(obj, handle);
        proxy.changename = 10;
        proxy.changename = "4";
        console.log(proxy.changename); 

写了个题目 供大家练习,感觉有面试题那味儿了,大家看看自己得出的结果是多少。

揭晓答案: 13。

讲解:proxy.changename = 10; 走proxy里的set, set里进行判断是一个数值,然后调用obj对象进行赋值,赋值的时候会触发obj的set拦截,判断大于5,ok最后赋值成10.

第二次 赋值字符串4 set判断 不通过就没下文了,默认return 一个undefined。

然后就是最后的显示了,显示的时候会触发proxy里的get 然后执行这段代码

target[name] += 1;

也就是

target[name] =target[name] +1;

target[name] +1里的target[name] ,就相当于get操作触发obj里的get,于是就会加一,10+1变成了11,然后是11加1。也就是十二。
然后return target[name]; 又试一次get 操作 就是12 +1 结果就变成了 13。

映射

介绍

Reflect 是一个内置的 JavaScript 对象,它提供了一组用于操作对象的方法。这些方法与对象的操作行为相对应,例如属性访问、函数调用、实例化等。

通俗来讲,我们可以使用这个对象,来进行日常的对象操作,比如取值,赋值等等一些操作,他同样可以完成并且有一些其他的优点,我们使用reflect一般都是搭配proxy使用。

比如像下面这样。

const obj = {
  foo: 42,
};

const value = Reflect.get(obj, 'foo');
console.log(value); // 输出: 42

优点

  1. 方便处理和调用目标对象的默认行为:Reflect 提供了与目标对象的默认行为相对应的方法,比如 Reflect.get、Reflect.set、Reflect.has 等。在 Proxy 的处理程序中使用 Reflect 可以方便地调用这些方法,从而实现对目标对象默认行为的处理和拦截。
  2. 更严谨的错误处理:Reflect 方法的返回值会更加准确地反映操作的结果,而不是像在使用目标对象的方法时可能会抛出异常。例如,Reflect.set 方法在赋值操作成功时会返回 true,失败时会返回 false,而不是抛出异常。
  3. 保持 Proxy 和目标对象的行为一致:Proxy 的目标对象可能是任意类型的对象,而 Reflect 提供了一致的方法,可以在 Proxy 的处理程序中对不同类型的目标对象进行操作,保持行为的一致性。
  4. 更清晰的代码逻辑:使用 Reflect 方法可以使代码更加简洁和易读,因为 Reflect 方法的命名和用法更加一致和直观,能够更清晰地表达代码的意图。

把它应用到我们上面的题。

 let obj = {
        name: 0,
        set changename(value) {
          if (value > 5) {
            this.name = value;
          }
        },
        get changename() {
          return (this.name += 1);
        },
      };
      let handle = {
        //target原对象,name是属性
        get: function (target, name) {
          target[name] += 1;
          //注意 get 一定要return
          return Reflect.get(target, name);
        },
        //value传的值
        set: function (target, name, value) {
          if (typeof value == "number") {
            Reflect.set(target, name, value)
          }
        },
      };
      let proxy = new Proxy(obj, handle);
      proxy.changename = 10;
      proxy.changename = "4";
      console.log(proxy.changename);
      ···

总结

对象属性的拦截(Object Property Interception):可以使用对象自身的 get 和 set 方法来拦截属性的获取和设置操作。通过在对象上定义这些方法,可以在属性访问过程中执行自定义的逻辑,例如验证、转换或拦截。

代理对象(Proxy Object):代理对象是一个代理目标对象的对象,通过代理对象可以拦截对目标对象的操作。代理对象使用 Proxy 构造函数创建,并提供了一组拦截器(handler)来定义拦截行为。

Reflect 映射(Reflect Mapping):Reflect 是一个内置的对象,提供了一组与对象操作相关的方法,用于执行与目标对象相同的操作。Reflect 方法提供了一种更简洁、易读的方式来执行常见的对象操作,如获取属性值、设置属性值、调用函数等。

到此这篇关于js对象属性的拦截与Proxy代理与Reflect映射的用法和区别讲解的文章就介绍到这了,更多相关js对象属性拦截内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • js的hasownproperty使用示例

    js的hasownproperty使用示例

    我们在js中可能经常会用到for in来遍历对象中的属性,当然for in中得到的属性,只能是可枚举的属性,for in的时候,它会把对象的属性(包括原型的属性)遍历一遍,看面看示例就明白了
    2014-03-03
  • JavaScript 学习初步 入门教程

    JavaScript 学习初步 入门教程

    看了《21天学习JavaScript》的视频教程,感觉就是语法好熟悉,和C++类似,和C#类似,和Java类似!幡然悔悟这些语言的语法都是类似的,数据类型,对象方法(或者称之为函数,由于我是个菜鸟所以也数不清楚到底是什么)。
    2010-03-03
  • JavaScript 浏览器对象模型BOM使用介绍

    JavaScript 浏览器对象模型BOM使用介绍

    这篇文章主要介绍了JavaScript 浏览器对象模型BOM使用介绍,需要的朋友可以参考下
    2015-04-04
  • 老生常谈js数据类型

    老生常谈js数据类型

    下面小编就为大家带来一篇老生常谈js数据类型。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08
  • javascript设计模式之鸭子类型和多态

    javascript设计模式之鸭子类型和多态

    这篇文章主要为大家介绍了javascript鸭子类型和多态,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助<BR>
    2022-01-01
  • 关于JavaScript中的关联数组分析

    关于JavaScript中的关联数组分析

    本篇文章小编为大家介绍,关于JavaScript中的关联数组分析。有需要的朋友可以参考一下
    2013-04-04
  • javascript XMLHttpRequest对象全面剖析

    javascript XMLHttpRequest对象全面剖析

    通过不必把Web页面寄送到服务器而实现数据传送,XMLHttpRequest对象为客户端与服务器之间提供了一种动态的交互能力。
    2010-04-04
  • 深入理解JavaScript系列(41):设计模式之模板方法详解

    深入理解JavaScript系列(41):设计模式之模板方法详解

    这篇文章主要介绍了深入理解JavaScript系列(41):设计模式之模板方法详解,模板方法(TemplateMethod)定义了一个操作中的算法的骨架,而将一些步骤延迟到子类中,模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤,需要的朋友可以参考下
    2015-03-03
  • 详解Javascript基础之循环

    详解Javascript基础之循环

    这篇文章主要为大家介绍了Javascript基础之循环,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2021-12-12
  • 在微信、支付宝、百度钱包实现点击返回按钮关闭当前页面和窗口的方法

    在微信、支付宝、百度钱包实现点击返回按钮关闭当前页面和窗口的方法

    这篇文章主要介绍了在微信、支付宝、百度钱包实现点击返回按钮关闭当前页面和窗口的方法,传统的window.close()是无效的,必须要使用它们的js代码才能关闭。下面小编给大家分享下代码,一起看看吧
    2016-08-08

最新评论