zoukankan      html  css  js  c++  java
  • es6之Proxy,Reflect

    Proxy 可以理解成,在目标对象之前架设一层“拦截”,外界对该对象的访问,都必须先通过这层拦截,因此提供了一种机制,可以对外界的访问进行过滤和改写。

    var proxy = new Proxy(target, handler);

    new Proxy()表示生成一个Proxy实例,target参数表示所要拦截的目标对象,handler参数也是一个对象,用来定制拦截行为。

    如果handler没有设置任何拦截,那就等同于直接通向原对象。

    var target = {};
    var handler = {};
    var proxy = new Proxy(target, handler);
    proxy.a = 'b';
    target.a // "b"

    上面代码中,handler是一个空对象,没有任何拦截效果,访问proxy就等同于访问target

    同一个拦截器函数,可以设置拦截多个操作。

    //原始对象储存真实数据
        let obj={
        time:'2017-03-11',
        name:'net',
        _r:123
      };
    
      let monitor=new Proxy(obj,{
        // 拦截对象属性的读取
        get(target,key){
          return target[key].replace('2017','2018')
        },
        // 拦截对象设置属性
        set(target,key,value){
          if(key==='name'){
            return target[key]=value;
          }else{
            return target[key];
          }
        },
        // 拦截key in object操作
        has(target,key){
          if(key==='name'){
            return target[key]
          }else{
            return false;
          }
        },
        // 拦截delete
        deleteProperty(target,key){
          if(key.indexOf('_')>-1){
            delete target[key];
            return true;
          }else{
            return target[key]
          }
        },
        // 拦截Object.keys,Object.getOwnPropertySymbols,Object.getOwnPropertyNames
        ownKeys(target){
          return Object.keys(target).filter(item=>item!='time')
        }
      });
    
    console.log(monitor.time);
    //2018-03-11
      monitor.time='2018';
      monitor.name='mukewang';
      console.log(monitor.time,monitor);
    //2018-03-11 Proxy {time: "2017-03-11", name: "net", _r: 123}
      console.log('name' in monitor,'time' in monitor);
    //true false
    console.log(Object.keys(monitor));
    // ["name", "_r"]

    Proxy 支持的拦截操作一览,一共 13 种。

    • get(target, propKey, receiver):拦截对象属性的读取,比如proxy.fooproxy['foo']
    • set(target, propKey, value, receiver):拦截对象属性的设置,比如proxy.foo = vproxy['foo'] = v,返回一个布尔值。
    • has(target, propKey):拦截propKey in proxy的操作,返回一个布尔值。
    • deleteProperty(target, propKey):拦截delete proxy[propKey]的操作,返回一个布尔值。
    • ownKeys(target):拦截Object.getOwnPropertyNames(proxy)Object.getOwnPropertySymbols(proxy)Object.keys(proxy)for...in循环,返回一个数组。该方法返回目标对象所有自身的属性的属性名,而Object.keys()的返回结果仅包括目标对象自身的可遍历属性。
    • getOwnPropertyDescriptor(target, propKey):拦截Object.getOwnPropertyDescriptor(proxy, propKey),返回属性的描述对象。
    • defineProperty(target, propKey, propDesc):拦截Object.defineProperty(proxy, propKey, propDesc)Object.defineProperties(proxy, propDescs),返回一个布尔值。
    • preventExtensions(target):拦截Object.preventExtensions(proxy),返回一个布尔值。
    • getPrototypeOf(target):拦截Object.getPrototypeOf(proxy),返回一个对象。
    • isExtensible(target):拦截Object.isExtensible(proxy),返回一个布尔值。
    • setPrototypeOf(target, proto):拦截Object.setPrototypeOf(proxy, proto),返回一个布尔值。如果目标对象是函数,那么还有两种额外操作可以拦截。
    • apply(target, object, args):拦截 Proxy 实例作为函数调用的操作,比如proxy(...args)proxy.call(object, ...args)proxy.apply(...)
    • construct(target, args):拦截 Proxy 实例作为构造函数调用的操作,比如new proxy(...args)

    Proxy 实例的方法 

    下面是上面这些拦截方法的详细介绍。

    get()

     //原始对象储存真实数据
        let obj={
        time:'2017-03-11',
        name:'net',
        _r:123
      };
    
      let monitor=new Proxy(obj,{
        // 拦截对象属性的读取
        get(target,key){
          return target[key].replace('2017','2018')
        }
     });
    console.log(monitor.time,monitor);  
    //2018-03-11 Proxy {time: "2017-03-11", name: "net", _r: 123}
    //只是读取显示的值变了,实际的值并没有改变

    set()

    //原始对象储存真实数据
        let obj={
        time:'2017-03-11',
        name:'net',
        _r:123
      };
    
      let monitor=new Proxy(obj,{
        // 拦截对象属性的读取
        get(target,key){
          return target[key].replace('2017','2018')
        },
        // 拦截对象设置属性
        set(target,key,value){
          if(key==='name'){
            return target[key]=value;
          }else{
            return target[key];
          }
        }
      });
    monitor.time='2018';
     console.log(monitor.time);
    //2017-03-11 set中设置不做改变
    monitor.name='sun menghua'; 
    console.log(monitor.time,monitor); 
    //2018-03-11 Proxy {time: "2017-03-11", name: "sun menghua", _r: 123}
    //因为get monitor.time发生了改变

    ownkeys()

    for...in循环也受到ownKeys方法的拦截。

    const obj = { hello: 'world' };
    const proxy = new Proxy(obj, {
      ownKeys: function () {
        return ['a', 'b'];
      }
    });
    
    for (let key in proxy) {
      console.log(key); // 没有任何输出
    }

    上面代码中,ownkeys指定只返回ab属性,由于obj没有这两个属性,因此for...in循环不会有任何输出。

    Reflect

    Reflect对象与Proxy对象一样,也是 ES6 为了操作对象而提供的新 API.

    Reflect对象的方法与Proxy对象的方法一一对应,只要是Proxy对象的方法,就能在Reflect对象上找到对应的方法。这就让Proxy对象可以方便地调用对应的Reflect方法,完成默认行为,作为修改行为的基础。也就是说,不管Proxy怎么修改默认行为,你总可以在Reflect上获取默认行为。

    Reflect对象一共有 13 个静态方法。(似曾相识的13种)

    • Reflect.apply(target, thisArg, args)
    • Reflect.construct(target, args)
    • Reflect.get(target, name, receiver)
    • Reflect.set(target, name, value, receiver)
    • Reflect.defineProperty(target, name, desc)
    • Reflect.deleteProperty(target, name)
    • Reflect.has(target, name)
    • Reflect.ownKeys(target)
    • Reflect.isExtensible(target)
    • Reflect.preventExtensions(target)
    • Reflect.getOwnPropertyDescriptor(target, name)
    • Reflect.getPrototypeOf(target)
    • Reflect.setPrototypeOf(target, prototype)

    Reflect.get(target, name, receiver)

    var myObject = {
      foo: 1,
      bar: 2,
      get baz() {
        return this.foo + this.bar;
      },
    }
    
    Reflect.get(myObject, 'foo') // 1
    Reflect.get(myObject, 'bar') // 2
    Reflect.get(myObject, 'baz') // 3

    如果receiver存在,则函数的this绑定receiver

    var myObject = {
      foo: 1,
      bar: 2,
      get baz() {
        return this.foo + this.bar;
      },
    };
    
    var myReceiverObject = {
      foo: 4,
      bar: 4,
    };
    
    Reflect.get(myObject, 'baz', myReceiverObject) // 8
    {
      let obj={
        time:'2017-03-11',
        name:'net',
        _r:123
      };
    
      console.log(Reflect.get(obj,'time')); //2017-03-11
      Reflect.set(obj,'name','sun menghua');
      console.log(obj);
    ////{time: "2017-03-11", name: "sun menghua", _r: 123}
      console.log(Reflect.has(obj,'name'));//true
    }
  • 相关阅读:
    python机器学习系列之环境搭建
    github 出现 Permission denied (publickey)的解决
    Eclipse添加Web和java EE插件
    GNU Make 学习系列一:怎样写一个简单的Makefile
    GTK+编程概述
    SpringBoot导出excel数据报错Could not find acceptable representation
    Postman测试上传MultipartFile文件
    idea 代码没有被svn控制
    FPGA基础学习(1) -- FFT IP核(Quartus)
    markdown使用小结
  • 原文地址:https://www.cnblogs.com/sunmarvell/p/9144199.html
Copyright © 2011-2022 走看看