ES6 Proxy和Reflect

1.Proxy是什么?

Proxy 对象用于创建一个对象的代理,从而实现基本操作的拦截和自定义(如属性查找、赋值、枚举、函数调用等)。属于“元编程”,即对编程语言进行编程。

如对对象a进行拦截就 let p = new Proxy(obj,handler)在handler里面编写拦截逻辑;

又如有对象a和b,想要对对象a中实现拦截对象b的属性,就将对象b设置到对象a的原型上再编写拦截逻辑;

2.创建Proxy对象的两种基本语法

  • 创建对象的代理
const p = new Proxy(target, handler)
  • 创建一个可撤销的代理对象
const { proxy: p, revoke } = Proxy.revocable(data, handler)

3.创建的proxy的实例有三个属性

  • target: 要使用 Proxy 包装的目标对象(可以是任何类型的对象,包括原生数组,函数,甚至另一个代理)。
  • handler: 一个通常以函数作为属性的对象,各属性中的函数分别定义了在执行各种操作时代理 p 的行为。(获取或者设置属性等操作;handler函数触发时,handler函数里面的this指向handler函数)
  • IsRevoked:当前属性是否被撤销。const { proxy: p, revoke } = Proxy.revocable(data, handler)创建一个可撤销的代理对象

4.proxy的实例两个注意点:

4.1 handler函数:handler函数触发时,handler函数里面的this指向handler函数

        let obj = { a:1 };
        let handler = {
            get(target, property, receiver){
                console.log(this);
                return target[property];//get中需要返回对应属性
            }
        };
        let p =  new Proxy(obj,handler);
        console.log(p.a);//注意要通过代理对象获取才能触发 get拦截

 

4.2 IsRevoked属性

IsRevoked:当前属性是否被撤销。const { proxy: p, revoke } = Proxy.revocable(data, handler)创建一个可撤销的代理对象

通过const { proxy: p, revoke } = Proxy.revocable(data, handler)创建可被撤销的对象实例后,此时IsRevoked为false,p.a 去调用属性会返回;

当在调用revoke();方法用于撤销代理对象,调用后IsRevoked为true,p.a 去调用属性会报错;

        let obj = { a:1 };
        let handler = {};
        const { proxy, revoke } = Proxy.revocable(obj, handler);
        

ES6 Proxy和Reflect_第1张图片

 5.handler中的方法handler.get()

var p = new Proxy(target, {
  get (target, property, receiver) {
  }
})

5.1方法用于拦截对象的读取属性操作。

  • target表示目标对象
  • property 被获取的属性名
  • receiver 接收Proxy或者继承Proxy对象 receiver === p
        let obj = { a: 1, b: 2 };
        let handler = {
            get(target, property, receiver) {
                console.log(target, property, receiver);
                return target[property];
            }
        };
        let proxy = new Proxy(obj, handler);
        console.log(proxy.a);

 

5.2 receiver 表示get()方法中接收的对象或者继承自的proxy对象

这里的问题是代理对象proxy.name想要获取到的是obj1上的name。

5.3handler.get()方法会拦截目标对象的以下操作:

  • 访问属性:proxy[foo] 和 proxy.bar
  • 访问原型链上的属性:Object.create(proxy)[foo]
  • Reflect.get()访问属性

访问属性:proxy[foo] 和 proxy.bar

比如,使用Proxy代理对象obj2,并将proxy实例对象设置到obj1的原型上,则通过obj1就可以获取到Proxy代理后的obj2的属性。此时通过proxy.name就可以获取到obj2的属性name值"lmf"

        let obj1 = { a: 1, b: 2, name: 'allen' };
        let obj2 = {
            name: 'lmf',
            get value() {
                return this.name
            }
        }
        let handler = {
            get(target, property, receiver) {
                return target[property];
                // return Reflect.get(target, property, receiver);
            }
        };
        let proxy = new Proxy(obj2, handler);
        // 将obj2的属性设置到obj1的原型上,则通过obj1就可以获取到obj2的属性
        Object.setPrototypeOf(obj1, proxy);
        console.log(obj1);
        console.log(obj1.value);//lmf

 其实这里obj1.name应该为allen才对,可以通过return Reflect.get(target, property, receiver);设置,类似与改变this指向为目标对象target也就是obj1

        let obj1 = { a: 1, b: 2, name: 'allen' };
        let obj2 = {
            name: 'lmf',
            get value() {
                return this.name
            }
        }
        let handler = {
            get(target, property, receiver) {
                return Reflect.get(target, property, receiver);
            }
        };
        let proxy = new Proxy(obj2, handler);
        // 将obj2的属性设置到obj1的原型上,则通过obj1就可以获取到obj2的属性
        Object.setPrototypeOf(obj1, proxy);
        console.log(obj1);
        console.log(obj1.value);//lmf

 访问原型链上的属性:Object.create(proxy)[foo]

        // 访问原型链上的属性:Object.create(proxy)[foo]
        let obj1 = { foo:'foo' };
        let handler = {
            get(target, property, receiver) {
                console.log("触发了");
                return target[property]
            }
        };
        let proxy = new Proxy(obj1, handler);
        const p = Object.create(proxy);
        console.log(p['foo']);

 Reflect.get(proxy, property)访问属性

        //  Reflect.get()访问属性
        let obj1 = { foo: 'foo' };
        let handler = {
            get(target, property, receiver) {
                console.log("触发了");
                return target[property]
            }
        };
        let proxy = new Proxy(obj1, handler);
        Reflect.get(proxy, 'foo')

6.handler中的方法handler.set()

 handler.set() 方法是设置属性值操作的捕获器。

const p = new Proxy(target, {
  set: function(target, property, value, receiver) {
  }
});

该方法会拦截目标对象的以下操作

1. 指定属性值:proxy[foo] = bar 和 proxy.foo = bar

2. 指定继承者的属性值:Object.create(proxy)[foo] = bar

3. Reflect.set()

        let obj1 = {
            myTime : "1697683657936"
        }
        let handler = {
            set(target, property, value){
                let setTime = value + new Date().getTime();
                target[property] = setTime;
            }
        }
        let proxy = new Proxy(obj1,handler);
        proxy.myTime = "此时的时间戳是:";
        console.log(proxy.myTime);

7.利用proxy实现的功能

7.1 验证对象的传值

需要实现的功能

1. 验证age属性的数据类型是否为整数

  2. 验证值的范围是否小于等于200

        // 需要实现的功能
        // 1. 验证age属性的数据类型是否为整数
        // 2. 验证值的范围是否小于等于200
        let user = {
            age: 100
        }

        let handler = {
            set(target,property,value){
                
                if(property==='age'){
                    if(!Number.isInteger(value)){
                        throw new TypeError("age的值必须是整数");
                    }
                    if(value>200){
                        throw new RangeError("age的值不能超过200");
                    }
                }
            }
        }

        let proxy = new Proxy(user,handler);
        // proxy.age = 22.3;
        proxy.age = 300

7.2通过属性查找数组中的特定对象

 需要实现的功能

var data = [
  { name: 'Firefox'    , type: 'browser' },
  { name: 'SeaMonkey'  , type: 'browser' },
  { name: 'Thunderbird', type: 'mailer' }
]

1. 通过索引返回对应数据 proxy[0]

2. 通过number属性返回数组长度 proxy.number

3. 通过name获取对应的数据 proxy['Firefox']

4. 通过type返回对应的数据 proxy['browser']

5. 通过types返回data中的type products.types

        var data = [
            { name: 'Firefox', type: 'browser' },
            { name: 'SeaMonkey', type: 'browser' },
            { name: 'Thunderbird', type: 'mailer' }
        ]
        let handler = {
            get(target, property) {
                // 2. 通过number属性返回数组长度 proxy.number
                if (property === 'number') {
                    return target.length;
                }

                let result = [];
                target.forEach((item, index) => {
                    // 1. 通过索引返回对应数据 proxy[0]
                    if (property == index) {
                        return result.push(item);
                    }

                    // 3. 通过name获取对应的数据 proxy['Firefox']
                    if (property === item.name) {
                        return result.push(item);
                    }

                    // 4. 通过type返回对应的数据 proxy['browser']
                    if (property === item.type) {
                        return result.push(item);
                    }
                    // 5. 通过types返回data中的type products.types
                    if (property === 'types') {
                        result.push(item.type);
                        result = [...new Set(result)]
                    }
                });
                return result;
            }
        }
        let proxy = new Proxy(data, handler);
        console.log(proxy[0]);
        console.log(proxy.number);
        console.log(proxy['Firefox']);
        console.log(proxy['browser']);
        console.log(proxy['types']);

你可能感兴趣的:(ES6高阶,Proxy和Reflect)