当前位置 主页 > 网站技术 > 代码类 >

    ES6之Proxy的get方法详解

    栏目:代码类 时间:2019-10-27 12:05

    Proxy是在ES2015(ES6)中新添加内置对象,用于自定义一些基本操作。

    这篇文章是我在学习Proxy的时候对于get方法的一些心得。

    作为ES2015新定义的内置对象,Proxy 能够拦截并且自定义对象以及函数的一些基本操作,具有很高的优先级和便利性,能够让我们在写代码的时候多出一种解决难题的途径。

    Proxy的get方法用于拦截对象属性的读取操作,例如 obj.key 和 obj.[key]。在给Proxy的handler参数中设置get方法后,每当进行读取操作时,会优先调用该get方法,我们可以在这个方法函数中对读取行为进行拦截。请看下面的代码:

    const obj = { key: 1 }
    const proxy = new Proxy(obj, {
     get: function(target, property, receiver) {
      console.log('get', property)
      return target[property]
     }
    })
    console.log(proxy.key)
    // get key
    // 1
    

    get方法的参数一共有三个:target是实例化Proxy时使用的对象,在这个例子中是obj;而property是这次读取操作中想要获取的属性名,在这个例子中是key;最后一个参数receiver则是这个实例化的Proxy自身,即proxy。
    在这个例子中,我在get方法的最后返回了target[property],这是为了能够让读取操作能够进行下去。由于Proxy的get方法是最先被调用的,所以这里返回的内容就是我们读取操作能够获得的结果;如果我们在这里不返回任何值,那么就会得到undefined。

    receiver和死循环

    要注意的是,千万不要在get方法中读取receiver的属性,因为receiver实质上就是proxy自身,所以receiver.key这句代码就等同于proxy.key,会重新调用get方法导致死循环。

    const obj = { key: 1 }
    const proxy = new Proxy(obj, {
     get: function(target, property, receiver) {
      console.log(receiver.key)
      return target[property]
     }
    })
    console.log(proxy.key)
    // 死循环!
    

    原型链上的getter

    有时候,我们会在对象之中使用getter和setter来定制属性的赋值和读取。在这时,如果proxy的get方法内部有使用到target[property]的话,target[property]的值会受到目标对象的getter的影响。因此调用get方法的时候请注意在目标对象中是否有用到getter。

    const obj = {
     get key() {
      return 'string'
     },
     set key(value) {
      console.log(`key is ${value}, it is a ${typeof value}`)
     }
    }
    const proxy = new Proxy(obj, {
     get: function(target, property, receiver) {
      if(typeof target[property] !== 'string') {
       return target[property]
      } else {
       throw new TypeError(`The type of ${property} is String!`)
      }
     }
    })
    proxy.key = 100
    console.log(proxy, obj)
    // key is 100, it is a number
    // The type of key is String!
    

    在上方的例子中,如果访问obj的非数字类型的属性,就会抛出一个错误,但是由于obj中getter的原因,无论我给key属性赋什么值,在访问key属性的时候肯定会抛出错误。

    如果仅仅要注意目标对象中的getter还算容易的,但是如果目标对象继承自其他对象,那么事情就变得有些麻烦了,请看下面的例子:

    const parentObj = {
     get key() {
      return 'string'
     },
     set key(value) {
      console.log(`key is ${value}, it is a ${typeof value}`)
     }
    }
    const obj = Object.create(parentObj)
    const proxy = new Proxy(obj, {
     get: function (target, property, receiver) {
      if (typeof target[property] !== 'string') {
       return target[property]
      } else {
       throw new TypeError(`The type of ${property} is String!`)
      }
     }
    })
    proxy.key = 100 
    console.log(proxy.key)
    // key is 100, it is a number
    // The type of key is String!