ES6 Proxies and Reflection

此文出处

简介

  • proxy
    proxy可以拦截目标(target)上的非内置的对象进行操作,使用trap拦截这些操作,trap是响应特定操作的方法。

  • reflection
    reflection是通过Reflect对象表示,他提供了一些方法集,为代理proxy提供默认行为。

下面是一些proxy trapReflect方法,每个proxy trap都有提供对应的Reflect方法,他们接收相同的参数

Proxy Trap Overrides the Behavior Of Default Behavior
get Reading a property value Reflect.get()
set Writing to a property Reflect.set()
has The in operator Reflect.has()
deleteProperty The delete operator Reflect.deleteProperty()
getPrototypeOf Object.getPrototypeOf() Reflect.getPrototypeOf()
setPrototypeOf Object.setPrototypeOf() Reflect.setPrototypeOf()
isExtensible Object.isExtensible() Reflect.isExtensible()
preventExtensions Object.preventExtensions() Reflect.preventExtensions()
getOwnPropertyDescriptor Object.getOwnPropertyDescriptor() Reflect.getOwnPropertyDescriptor()
defineProperty Object.defineProperty() Reflect.defineProperty
ownKeys Object.keys, Object.getOwnPropertyNames(), Object.getOwnPropertySymbols() Reflect.ownKeys()
apply Calling a function Reflect.apply()
construct Calling a function with new Reflect.construct()

这里的每个trap都会覆盖对象的内置行为,便于拦截和修改对象。如果你真的需要内置行为,可以使用相对应的Reflect方法。

开始的时候,ES6的规范有个enumerate trap,用于改变for..inObject.keys的枚举属性,但是在实行的时候发现有些困难,于是在ES7中移除了。所以这里不讨论他。

创建一个简单的代理

当你使用Proxy的构造函数去创建代理的时候,他接受两个参数,一个是目标对象(target),另外一个是处理对象(handler)。这个handler定义了一个或者多个trap去处理代理,如果没有定义trap,那么就会使用默认的行为。

let target = {};

let proxy = new Proxy(target, {});

proxy.name = "proxy";
console.log(proxy.name);        // "proxy"
console.log(target.name);       // "proxy"

target.name = "target";
console.log(proxy.name);        // "target"
console.log(target.name);       // "target"

从上面这个例子可以发现,不管是proxy或者是target的属性更改,都会影响到另外一个。其实这就是这两个的关系: proxy本身不存储这个属性,他只是把操作转发到target上面的这个例子似乎没啥意思,并没有体现出核心trap的价值所在。

使用set trap验证属性

set trap是在设置属性值时触发。
set trap接收这几个参数:

  1. trapTarget - 接收的属性的对象,就是代理的目标。
  2. key - 要写入的属性的key(string || symbol)
  3. value - 写入属性的值
  4. receiver - 操作的对象,通常是代理

Reflect.setset trap相对应的方法。如果属性被设置,那么trap应该返回true,如果没有被设置,那就返回falseReflect.set()会根据操作是否成功返回正确的值。

要验证一个属性的值,那就需要使用set trap来检查这个值,看下面代码:

let target = {
    name: "target"
};

let proxy = new Proxy(target, {
    set(trapTarget, key, value, receiver) {
        console.log(`trapTarget is ${trapTarget}, key is ${key}, value is ${value}, receiver is ${receiver}`)
        // 忽视存在的属性,以免产生影响
        if (!trapTarget.hasOwnProperty(key)) {
            if (isNaN(value)) {
                throw new TypeError("Property must be a number.");
            }
        }

        // 添加到属性
        return Reflect.set(trapTarget, key, value, receiver);
    }
});

// 添加一个新的属性
proxy.count = 1;
console.log(proxy.count);       // 1
console.log(target.count);      // 1

// 赋值给存在target上的属性
proxy.name = "proxy";
console.log(proxy.name);        // "proxy"
console.log(target.name);       // "proxy"

// 新的属性值不是数字会抛出异常
proxy.anotherName = "proxy";

可以发现,每次设置属性值的时候都会进行拦截判断,所以,相对的,你在获取的时候,可以使用get进行拦截判断。

使用get trap验证

js一个有趣又令人困惑的地方就是获取一个不存在的属性的时候,不会抛出异常,只会返回一个undefined。不像其他的大多数语言,都会抛出一个错误,可能你写了大量的代码,你可能会意识到这是一个严重的问题,比如拼写错误的这些问题,代理可以为你处理这些。

get是在读取对象属性的时候用到的trap他接收三个参数:

  1. trapTarget - 从哪个对象读取的属性,就是target.
  2. key - 读取的key
  3. receiver - 操作的对象,通常是代理(proxy)
    可以发现这个和上面的set差不多,就是少了一个设置的value参数。相对的,Reflect.get方法接受与get trap相同的三个参数,并返回属性的默认值。
var proxy = new Proxy({}, {
        get(trapTarget, key, receiver) {
            if (!(key in receiver)) {
                throw new TypeError("Property " + key + " doesn't exist.");
            }

            return Reflect.get(trapTarget, key, receiver);
        }
    });

proxy.name = "proxy";
console.log(proxy.name);            // "proxy"

// 不存在这个属性,抛出错误
console.log(proxy.age);             // throws error

不知道你有没有发现,我们在这里使用receiver代替trapTarget配合in一起使用,我们将在下面介绍。

使用has trap隐藏属性的存在

in这个操作想来大家比较熟悉的,是确定属性是否存在对象及原型链上。

var target = {
    value: 42;
}

console.log("value" in target);     // true
console.log("toString" in target);  // true

代理允许你使用has这个trap去返回不同的值。
这个has trap是在使用in操作时触发。has trap接收两个参数:

  1. trapTarget
  2. key
    Reflect.has方法接受这些相同的参数并返回in运算符的默认响应。使用has trapReflect.has可以改变某些属性的in行为,同时又回退到其他属性的默认行为。例如你只想隐藏value属性:
var target = {
    name: "target",
    value: 42
};

var proxy = new Proxy(target, {
    has(trapTarget, key) {

        if (key === "value") {
            return false;
        } else {
            return Reflect.has(trapTarget, key);
        }
    }
});


console.log("value" in proxy);      // false
console.log("name" in proxy);       // true
console.log("toString" in proxy);   // true

可以发现上例直接判断,如果不是value key,就使用Reflect去返回其默认行为。

使用deleteProperty trap对删除进行操作

通过属性描述那部分我们知道,delete是通过configurable来控制的,非严格模式下删除会返回false,严格模式下会报错。但是我们可以使用代理deleteProperty trap去操作他这个行为。

下面我们再来看看deleteProperty这个trap。他也是接受两个参数:

  1. trapTarget
  2. key
    Reflect.deleteProperty方法提供了deleteProperty trap相对的行为去实现。所以我们可以使用这两个去改变delete的默认行为。
let target = {
    name: "target",
    value: 42
};

let proxy = new Proxy(target, {
    deleteProperty(trapTarget, key) {

        if (key === "value") {
            return false;
        } else {
            return Reflect.deleteProperty(trapTarget, key);
        }
    }
});

// Attempt to delete proxy.value

console.log("value" in proxy);      // true

let result1 = delete proxy.value;
console.log(result1);               // false

console.log("value" in proxy);      // true

// Attempt to delete proxy.name

console.log("name" in proxy);       // true

let result2 = delete proxy.name;
console.log(result2);               // true

console.log("name" in proxy);       // false

这样可以拦截操作,好奇的你可能会想去操作nonconfigurable的时候,也可以删除,你可以尝试一下。这个方法在受保护的属性被删除的时候,非严格模式下会抛错。

原型的代理trap

这个章节里介绍了setPrototypeOfgetPrototypeOf。代理也为这两种情况添加了相应的trap。针对这两个代理的trap,都有不同的参数:

  • setPrototypeOf
    1. trapTarget
    2. proto 这个用作原型的对象
      他和Reflect.setPrototypeOf接收的参数相同,去做相对应的操作。另一方面,getPrototypeOf只接收一个参数trapTarget,相应的也存在Reflect.getPrototypeOf方法.
原型代理是如何工作的

他们有一些限制。首先,getPrototypeOf只可以返回对象或者null,返回其他的,在运行的时候会报错。同样的,setPrototypeOf trap如果失败,必须返回false,并且Object.setPrototypeOf会抛错, 如果返回其他的值,那就是认为操作成功。
下面来看一个例子:

var target = {};
var proxy = new Proxy(target, {
    getPrototypeOf(trapTarget) {
        return null;
    },
    setPrototypeOf(trapTarget, proto) {
        return false;
    }
});

var targetProto = Object.getPrototypeOf(target);
var proxyProto = Object.getPrototypeOf(proxy);

console.log(targetProto === Object.prototype);      // true
console.log(proxyProto === Object.prototype);       // false
console.log(proxyProto);                            // null

// succeeds
Object.setPrototypeOf(target, {});

// throws error
Object.setPrototypeOf(proxy, {});

从上面可以发现,对于proxy进行了拦截,以至于原型不同。然后对proxy进行setPrototypeOf会抛出异常,就是上面提到的,setPrototypeOf返回false,那么Object.setPrototypeOf会抛出错误。
当然,如果你想要使用它的默认行为,那就需要使用Reflect对象的方法来操作。

为什么会有两套方法

让人感到困惑的是, setPrototypeOf trapgetPrototypeOf trap看起来和Object.getPrototypeOf() or Object.setPrototypeOf()看起来类似,为什么还要这两套方法。其实他们看起来是类似,但是还有很大的差异:
首先,Object.getPrototype||Object.setPrototypeOf在一开始就是为开发人员创建的高级别的操作。然而Reflect.getPrototypeOf || Reflect.setPrototypeOf是提供了操作以前仅仅用于内部的[[GetPrototypeOf]] || [[SetPrototypeOf]]的底层属性。Reflect.getPrototypeOf方法是内部[[GetPrototypeOf]]操作的包装器(带有一些输入验证)。Reflect.setPrototypeOf方法和[[SetPrototypeOf]]具有相同的关系。Object上的相应方法也调用[[GetPrototypeOf]][[SetPrototypeOf]],但在调用之前执行几个步骤并检查返回值以确定如何操作。

上面说的比较泛泛,下面来详细说下:
如果Reflect.getPrototypeOf方法的参数不是对象或者null,则抛出错误;而Object.getPrototypeOf在执行操作之前首先将值强制转换为对象。

var result1 = Object.getPrototypeOf(1);
console.log(result1 === Number.prototype);  // true

// throws an error
Reflect.getPrototypeOf(1);

很明显,Object上的可以工作,他把数字1转换成了对象,Reflect上的不会进行转换,所以抛出异常。

setPrototypeOf也有一些不同,Reflect.setPrototypeOf会返回一个布尔来确定是否成功,false就是失败。然而Object.setPrototypeOf如果失败,会抛出错误。

对象 Extensibility trap

ECMAScript 5通过Object.preventExtensionsObject.isExtensible方法添加了对象可扩展性的操作,因此ES6在此基础上对这两个方法添加了代理。并且这两个代理方法都只接收一个参数trapTarget.isExtensible trap必须返回布尔值来确定是否是可扩展的,preventExtensions trap返回布尔值确定是否成功。
Reflect对象里的这两个方法都会返回布尔值,所以这两个是可以作为相对应的方法去使用实现默认行为。

两个简单的例子
var target = {};
var proxy = new Proxy(target, {
    isExtensible(trapTarget) {
        return Reflect.isExtensible(trapTarget);
    },
    preventExtensions(trapTarget) {
        return Reflect.preventExtensions(trapTarget);
    }
});


console.log(Object.isExtensible(target));       // true
console.log(Object.isExtensible(proxy));        // true

Object.preventExtensions(proxy);

console.log(Object.isExtensible(target));       // false
console.log(Object.isExtensible(proxy));        // false

这个例子就是使用代理拦截并返回他的默认行为,等于默认的情况。所以经过Object属性操作之后,就是返回默认的行为。

如果我们不想他拓展,我们可以这样去处理:

var target = {};
var proxy = new Proxy(target, {
    isExtensible(trapTarget) {
        return Reflect.isExtensible(trapTarget);
    },
    preventExtensions(trapTarget) {
        return false
    }
});


console.log(Object.isExtensible(target));       // true
console.log(Object.isExtensible(proxy));        // true

Object.preventExtensions(proxy);

console.log(Object.isExtensible(target));       // true
console.log(Object.isExtensible(proxy));        // true

这里他不会成功,因为返回了false,没有使用对应的Reflect去做相对的默认行为处理,所以操作不会转发到操作的目标。

Duplicate Extensibility Methods

如果传递对象值作为参数,方法Object.isExtensibleReflect.isExtensible类似。如果不是对象作为参数传递,Object.isExtensible始终返回false,而Reflect.isExtensible则抛出错误。

let result1 = Object.isExtensible(2);
console.log(result1);                       // false

// throws error, Reflect.isExtensible called on non-object
let result2 = Reflect.isExtensible(2);

这个限制类似于Object.getPrototypeOfReflect.getPrototypeOf方法之间的差异,因为具有较低级别功能的方法具有比其更高级别对应方更严格的错误检查。

Object.preventExtensionsReflect.preventExtensions方法也非常相似。 Object.preventExtensions方法始终返回作为参数传递给它的值,即使该值不是对象也是如此。然而另一方面,如果参数不是对象,那么Reflect.preventExtensions方法会抛出错误;如果参数是一个对象,那么Reflect.preventExtensions在操作成功时返回true,否则返回false

var result1 = Object.preventExtensions(2);
console.log(result1);                               // 2

var target = {};
var result2 = Reflect.preventExtensions(target);
console.log(result2);                               // true

// throws error
var result3 = Reflect.preventExtensions(2);

这个例子就是对上面的总结。

Property Descriptor Traps

ECMAScript 5最重要的功能之一是使用Object.defineProperty方法定义属性具体属性的能力。在以前的JavaScript版本中,无法定义访问者属性,使属性成为只读,或使属性不可数。具体参考这里

代理允许分别使用defineProperty trapgetOwnPropertyDescriptor trap拦截对Object.definePropertyObject.getOwnPropertyDescriptor的调用。 defineProperty trap接收以下参数:

  1. trapTarget - 被定义属性的对象(代理的目标)
  2. key
  3. descriptor
    defineProperty trap返回布尔值。getOwnPropertyDescriptor trap只接收trapTargetkey,并且返回描述信息。相应的Reflect.definePropertyReflect.getOwnPropertyDescriptor方法接受与其代理trap对应方相同的参数。
    例如:
var proxy = new Proxy({}, {
    defineProperty(trapTarget, key, descriptor) {
        return Reflect.defineProperty(trapTarget, key, descriptor);
    },
    getOwnPropertyDescriptor(trapTarget, key) {
        return Reflect.getOwnPropertyDescriptor(trapTarget, key);
    }
});


Object.defineProperty(proxy, "name", {
    value: "proxy"
});

console.log(proxy.name);            // "proxy"

var descriptor = Object.getOwnPropertyDescriptor(proxy, "name");

console.log(descriptor.value);      // "proxy"

很简单的一个例子,基本没有在拦截上做任何操作,只是返回他的默认行为。

Blocking Object.defineProperty()

trap返回true时,Object.defineProperty表示成功;
trap返回false时,Object.defineProperty会抛出错误。
可以使用这个功能来限制Object.defineProperty方法可以定义的属性类型.如下:

var proxy = new Proxy({}, {
    defineProperty(trapTarget, key, descriptor) {
        if (typeof key === "symbol") {
            return false;
        }

        return Reflect.defineProperty(trapTarget, key, descriptor);
    }
});


Object.defineProperty(proxy, "name", {
    value: "proxy"
});

console.log(proxy.name);                    // "proxy"

var nameSymbol = Symbol("name");

// throws error
Object.defineProperty(proxy, nameSymbol, {
    value: "proxy"
});

这里我们检测key的类型,如果是symbol就返回false.对于Object.defineProperty,返回false会抛出异常。

当然可以通过返回true而不调用Reflect.defineProperty方法使Object.defineProperty默认是失败的,这就避免错误的抛出。

Descriptor Object Restrictions

为了确保在使用Object.definePropertyObject.getOwnPropertyDescriptor方法时的一致行为,传递给defineProperty trap的描述符对象被规范化。从getOwnPropertyDescriptor trap返回的对象总是出于同样的原因进行验证。

不管哪个参数作为第三个参数传递给Object.defineProperty方法,都只能是下面这几种:enumerable, configurable, value, writable, get, set 这些将被作为descriptor传递。例如:

var proxy = new Proxy({}, {
    defineProperty(trapTarget, key, descriptor) {
        console.log(descriptor.value);              // "proxy"
        console.log(descriptor.name);               // undefined
        console.log(descriptor.writable)          // undefined
        return Reflect.defineProperty(trapTarget, key, descriptor);
    }
});


Object.defineProperty(proxy, "name", {
    value: "proxy",
    name: "custom"
});

可以发现,name不存在那几个descriptor里,所以传递不进去,不接收。并且这个和Object.defineProperty不同,没有进行一些包装,不存在默认的writable, configurable这些..。但是按理来说,你传递一个对象进行,他就应该接收啊,为啥这里会是undefined呢?这是因为descriptor实际上不是对传递给Object.defineProperty方法的第三个参数的引用,而是一个仅包含允许属性的新对象。Reflect.defineProperty方法还会忽略描述符上的任何非标准属性

getOwnPropertyDescriptor稍微有些不同,他会返回null, undefined,object.如果返回的是对象,那么对象只会包含上面可能出现的descriptor的这几种情况。

如果返回具有不允许的属性的对象,会导致错误,如下代码:

var proxy = new Proxy({}, {
    getOwnPropertyDescriptor(trapTarget, key) {
        return {
            name: "proxy"
        };
    }
});

// throws error
var descriptor = Object.getOwnPropertyDescriptor(proxy, "name");

因为name不属于descriptor接受的范围,所以引发了错误。这个限制可确保Object.getOwnPropertyDescriptor返回的值始终具有可靠的结构,无论代理使用什么。

Duplicate Descriptor Methods

和上面的一些trap类似,这个也有一些让人为之困惑的类似的方法。这里的是Object.defineProperty&Object. getOwnPropertyDescriptorReflect. defineProperty&Reflect.getOwnPropertyDescriptor.

defineProperty() Methods

看看这个方法的异同.
Object.definePropertyReflect.defineProperty方法完全相同,只是它们的返回值有所不同。

var target = {};

var result1 = Object.defineProperty(target, "name", { value: "target "});

console.log(target === result1);        // true

var result2 = Reflect.defineProperty(target, "name", { value: "reflect" });

console.log(result2);                   // true

可以发现,Object.defineProperty返回的是传入的第一个参数,Reflect.defineProperty返回的布尔值确定是否成功。

getOwnPropertyDescriptor() Methods

Object.getOwnPropertyDescriptor方法传入的参数是原始值的时候,会转换成对象进行处理。至于Reflect.getOwnPropertyDescriptor传入的不是对象,会抛出错误:

descriptor1 = Object.getOwnPropertyDescriptor(2, "name");
console.log(descriptor1);       // undefined

// throws an error
descriptor2 = Reflect.getOwnPropertyDescriptor(2, "name");

The ownKeys Trap

ownKeys trap允许你拦截内部的方法[[OwnPropertyKeys]]并覆盖默认的行为通过返回一组值。返回的这个数组值用于四个方法:Object.getOwnPropertyNames, Object.keys,Object.getOwnPropertySymbols()Object.assign(Object.assign用于数组来确定要复制的属性)。
ownKeys trap的默认行为是通过Reflect.ownKeys来实现,返回的就是一个数组,里面包含所有的属性keys(strings, symbols).
我们知道Object.keysObject.getOwnPropertyNames返回的是过滤掉symbol key的集合,但是Object.getOwnPropertySymbols却是相反,所以ownKeys集合了这几个之后,就可以返回所有的keys.并且Object.assign作用于stringssymbols键的对象。

ownKeys trap接收一个参数,就是trapTarget。他总是返回数组或者类似数组的值,否则会引发错误。
看下面这个例子:

var proxy = new Proxy({}, {
    ownKeys(trapTarget) {
        return Reflect.ownKeys(trapTarget).filter(key => {
            return typeof key !== "string" || key[0] !== "_";
        });
    }
});

var nameSymbol = Symbol("name");

proxy.name = "proxy";
proxy._name = "private";
proxy[nameSymbol] = "symbol";

var names = Object.getOwnPropertyNames(proxy),
    keys = Object.keys(proxy);
    symbols = Object.getOwnPropertySymbols(proxy);

console.log(names.length);      // 1
console.log(names[0]);          // "name"

console.log(keys.length);      // 1
console.log(keys[0]);          // "name"

console.log(symbols.length);    // 1
console.log(symbols[0]);        // "Symbol(name)"

最终返回的这个数组就是得到的结果。

ownKeys trap也会影响for-in循环,该循环调用trap来确定在循环内使用哪些键。

Function Proxies with the apply and construct Traps

这个可能是比较特殊的了。在代理的所有的trap中,只有apply trapconstruct trap这两个要求代理的target是必须一个function,我们知道function有两个内部的属性[[Call]][[Construct]]分别用于直接调用和new关键字调用的时候。因此apply trap在拦截直接调用的时候用到的,construct trap是拦截new调用时候用到的。

我们先来看看直接调用的的时候,

  • apply trap的参数:
    1. trapTarget
    2. thisArg - 调用期间的上下文对象this
    3. argumentsList - 传递到方法的数组参数

再来看看new关键字调用时候。

  • construct trap的参数
    1. trapTarget
    2. argumentsList

Reflect.construct方法也接受这两个参数,并有一个名为newTarget的可选第三个参数。如果给定这个第三个参数,newTarget这个参数就是new.target的值。

使用applyconstruct两个trap就可以拦截所有的方法调用.

var target = function() { return 42 },
var proxy = new Proxy(target, {
        apply: function(trapTarget, thisArg, argumentList) {
            return Reflect.apply(trapTarget, thisArg, argumentList);
        },
        construct: function(trapTarget, argumentList) {
            return Reflect.construct(trapTarget, argumentList);
        }
    });

// a proxy with a function as its target looks like a function
console.log(typeof proxy);                  // "function"

console.log(proxy());                       // 42

var instance = new proxy();
console.log(instance instanceof proxy);     // true
console.log(instance instanceof target);    // true

这个和上面几个类似,都是拦截之后使用它的默认行为。

Validating Function Parameters

下面来一个验证参数类型的例子:

// adds together all arguments
function sum(...values) {
    return values.reduce((previous, current) => previous + current, 0);
}

var sumProxy = new Proxy(sum, {
        apply(trapTarget, thisArg, argumentList) {

            argumentList.forEach((arg) => {
                if (typeof arg !== "number") {
                    throw new TypeError("All arguments must be numbers.");
                }
            });

            return Reflect.apply(trapTarget, thisArg, argumentList);
        },
        construct(trapTarget, argumentList) {
            throw new TypeError("This function can't be called with new.");
        }
    });

console.log(sumProxy(1, 2, 3, 4));          // 10

// throws error
console.log(sumProxy(1, "2", 3, 4));

// also throws error
var result = new sumProxy();

在这里,我们对参数进行了过滤处理,并且在new调用的时候,直接抛出错误,不让他去new

Calling Constructors Without new

我们之前介绍了关于new的相关介绍,判断一个函数是不是new调用,需要使用new.target来判断。

function Numbers(...values) {

    if (typeof new.target === "undefined") {
        throw new TypeError("This function must be called with new.");
    }

    this.values = values;
}

let instance = new Numbers(1, 2, 3, 4);
console.log(instance.values);               // [1,2,3,4]

// throws error
Numbers(1, 2, 3, 4);

可以发现,这个类似于在上面提到的使用proxy验证,但是这个明显更加方便一点。如果只是为了判断是否new调用,这个是可取的,但是有时候你需要知道做更多的控制,这个就办不到了。

function Numbers(...values) {

    if (typeof new.target === "undefined") {
        throw new TypeError("This function must be called with new.");
    }

    this.values = values;
}


let NumbersProxy = new Proxy(Numbers, {
        apply: function(trapTarget, thisArg, argumentsList) {
            return Reflect.construct(trapTarget, argumentsList);
        }
    });


let instance = NumbersProxy(1, 2, 3, 4);
console.log(instance.values);               // [1,2,3,4]

可以发现这个,在函数内部还是有检查,但是在表面调用的时候是没有使用这个new的,只是我们在代理里的apply trap里使用了Reflect.construct

Overriding Abstract Base Class Constructors

可以在Reflect.construct内传入第三个参数,用作new.target的值。这可以在构造函数中检查new.target的值的时候用到。

class AbstractNumbers {

    constructor(...values) {
        if (new.target === AbstractNumbers) {
            throw new TypeError("This function must be inherited from.");
        }

        this.values = values;
    }
}

class Numbers extends AbstractNumbers {}

let instance = new Numbers(1, 2, 3, 4);
console.log(instance.values);           // [1,2,3,4]

// throws error
new AbstractNumbers(1, 2, 3, 4);

上面可以发现有个限制,下面我们来试试使用代理来跳过.

class AbstractNumbers {

    constructor(...values) {
        if (new.target === AbstractNumbers) {
            throw new TypeError("This function must be inherited from.");
        }

        this.values = values;
    }
}

var AbstractNumbersProxy = new Proxy(AbstractNumbers, {
        construct: function(trapTarget, argumentList) {
            return Reflect.construct(trapTarget, argumentList, function() {});
        }
    });


let instance = new AbstractNumbersProxy(1, 2, 3, 4);
console.log(instance.values);               // [1,2,3,4]

这样,添加了第三个参数,这样new.target的值就是一个另外一个值了(匿名函数)。

Callable Class Constructors

我们知道class只能被new去调用,这是因为在构造函数的内部方法,[[Call]]被指定抛出错误。但是我们使用代理可以拦截这个内部属性,所以可以改变我们的调用方法。
例如我们想不通过new来调用一个class,可以通过代理,如下:

class Person {
    constructor(name) {
        this.name = name;
    }
}

var PersonProxy = new Proxy(Person, {
        apply: function(trapTarget, thisArg, argumentList) {
            return new trapTarget(...argumentList);
        }
    });


var me = PersonProxy("Nicholas");
console.log(me.name);                   // "Nicholas"
console.log(me instanceof Person);      // true
console.log(me instanceof PersonProxy); // true

可以发现,我们在apply这个trap对他进行了new一个。

Revocable Proxies

通常情况下,绑定了代理之后都是没有办法撤掉的,但是这个可以取消,通过Proxy.revocable去取消。这个方法和Proxy的构造函数传参类似,一个target和一个handler.
他返回的对象是有两个属性:

  • proxy - 被撤销的代理对象
  • revoke - 调用撤销代理的函数

revoke被调用的时候,就不能继续使用代理了。

var target = {
    name: "target"
};

var { proxy, revoke } = Proxy.revocable(target, {});

console.log(proxy.name);        // "target"

revoke();

// throws error
console.log(proxy.name);

可以发现,在调用revoke方法之后,代理就不能继续使用了。如果调用,就会抛出错误,不会返回undefined

Solving the Array Problem

看一个关于数组的问题:

let colors = ["red", "green", "blue"];

console.log(colors.length);         // 3

colors[3] = "black";

console.log(colors.length);         // 4
console.log(colors[3]);             // "black"

colors.length = 2;

console.log(colors.length);         // 2
console.log(colors[3]);             // undefined
console.log(colors[2]);             // undefined
console.log(colors[1]);             // "green"

在这里,length控制着数组的数据,一般情况下,我们没法子修改这些高级操作。

Detecting Array Indices

ECMAScript 6规范提供了有关如何确定属性键是否为数组索引的说明:

当且仅当ToString(ToUint32(P))等于PToUint32(p)不等于2的32次方减1时,字符串属性名P才是数组索引。

这个规范,在js中可以实现:

function toUint32(value) {
    return Math.floor(Math.abs(Number(value))) % Math.pow(2, 32);
}

function isArrayIndex(key) {
    let numericKey = toUint32(key);
    return String(numericKey) == key && numericKey < (Math.pow(2, 32) - 1);
}

toUint32函数使用规范中描述的算法将给定值转换为无符号的32位整数,isArrayIndex函数首先将密钥转换为uint32,然后执行比较以确定密钥是否为数组索引。

Increasing length when Adding New Elements

可以发现数组的行为,其实使用set trap就可以完成这两个行为。

function toUint32(value) {
    return Math.floor(Math.abs(Number(value))) % Math.pow(2, 32);
}

function isArrayIndex(key) {
    let numericKey = toUint32(key);
    return String(numericKey) == key && numericKey < (Math.pow(2, 32) - 1);
}

function createMyArray(length=0) {
    return new Proxy({ length }, {
        set(trapTarget, key, value) {

            let currentLength = Reflect.get(trapTarget, "length");

            // the special case
            if (isArrayIndex(key)) {
                let numericKey = Number(key);

                if (numericKey >= currentLength) {
                    Reflect.set(trapTarget, "length", numericKey + 1);
                }
            }

            // always do this regardless of key type
            return Reflect.set(trapTarget, key, value);
        }
    });
}

var colors = createMyArray(3);
console.log(colors.length);         // 3

colors[0] = "red";
colors[1] = "green";
colors[2] = "blue";

console.log(colors.length);         // 3

colors[3] = "black";

console.log(colors.length);         // 4
console.log(colors[3]);             // "black"

可以发现,上面对写入的key进行了验证。如果符合规范,则会给length进行添加操作。其他的会一直操作key.
现在,基于数组的length的第一个功能成立了,接下来是进行第二步。

Deleting Elements on Reducing length

这里就需要对减少的长度的部分进行删除了。

function toUint32(value) {
    return Math.floor(Math.abs(Number(value))) % Math.pow(2, 32);
}

function isArrayIndex(key) {
    let numericKey = toUint32(key);
    return String(numericKey) == key && numericKey < (Math.pow(2, 32) - 1);
}

function createMyArray(length=0) {
    return new Proxy({ length }, {
        set(trapTarget, key, value) {

            let currentLength = Reflect.get(trapTarget, "length");

            // the special case
            if (isArrayIndex(key)) {
                let numericKey = Number(key);

                if (numericKey >= currentLength) {
                    Reflect.set(trapTarget, "length", numericKey + 1);
                }
            } else if (key === "length") {

                if (value < currentLength) {
                    for (let index = currentLength - 1; index >= value; index\
--) {
                        Reflect.deleteProperty(trapTarget, index);
                    }
                }

            }

            // always do this regardless of key type
            return Reflect.set(trapTarget, key, value);
        }
    });
}

let colors = createMyArray(3);
console.log(colors.length);         // 3

colors[0] = "red";
colors[1] = "green";
colors[2] = "blue";
colors[3] = "black";

console.log(colors.length);         // 4

colors.length = 2;

console.log(colors.length);         // 2
console.log(colors[3]);             // undefined
console.log(colors[2]);             // undefined
console.log(colors[1]);             // "green"
console.log(colors[0]);             // "red"

可以发现,我们在每次length操作的时候,都会进行一次监听操作,用来减去他删除的部分。

Implementing the MyArray Class

使用代理创建类的最简单方法是像往常一样定义类,然后从构造函数返回代理。这样,实例化类时返回的对象将是代理而不是实例。(实例是构造函数内部的this值)。实例成为代理的目标,并返回代理,就好像它是实例一样。那么这个实例将完全私有,无法直接访问它,但可以通过代理间接访问它。
看一个简单的例子:

class Thing {
    constructor() {
        return new Proxy(this, {});
    }
}

var myThing = new Thing();
console.log(myThing instanceof Thing);      // true

我们知道,constructor内返回的基本数据类型不会影响他的返回,如果是非基本类型,那么就是具体的返回对象了。所以这里返回到是proxy,因此这里的这个myThing就是这个proxy. 由于代理会把他的行为传递给他的目标,因此myThing仍然被当做是Thing的实例。

考虑到上面这一点,使用代理创建自定义数组类相对简单点。代码与“删除减少长度的元素”部分中的代码大致相同。使用相同的代理代码,但这一次,它在类构造函数中。

function toUint32(value) {
    return Math.floor(Math.abs(Number(value))) % Math.pow(2, 32);
}

function isArrayIndex(key) {
    let numericKey = toUint32(key);
    return String(numericKey) == key && numericKey < (Math.pow(2, 32) - 1);
}

class MyArray {
    constructor(length=0) {
        this.length = length;

        return new Proxy(this, {
            set(trapTarget, key, value) {

                let currentLength = Reflect.get(trapTarget, "length");

                // the special case
                if (isArrayIndex(key)) {
                    let numericKey = Number(key);

                    if (numericKey >= currentLength) {
                        Reflect.set(trapTarget, "length", numericKey + 1);
                    }
                } else if (key === "length") {

                    if (value < currentLength) {
                        for (let index = currentLength - 1; index >= value; i\
ndex--) {
                            Reflect.deleteProperty(trapTarget, index);
                        }
                    }

                }

                // always do this regardless of key type
                return Reflect.set(trapTarget, key, value);
            }
        });

    }
}


let colors = new MyArray(3);
console.log(colors instanceof MyArray);     // true

console.log(colors.length);         // 3

colors[0] = "red";
colors[1] = "green";
colors[2] = "blue";
colors[3] = "black";

console.log(colors.length);         // 4

colors.length = 2;

console.log(colors.length);         // 2
console.log(colors[3]);             // undefined
console.log(colors[2]);             // undefined
console.log(colors[1]);             // "green"
console.log(colors[0]);             // "red"

这就是利用我们上面提到的那点,返回的最终是个代理来完成这个操作。

尽管这样很容易,但是他为每一个新的实例都创建了一个代理。但是我们可以为每一个实例都共享一个代理,那就是通过原型。

Using a Proxy as a Prototype

代理可以用作原型,但是这样会提高复杂度,比上面的实现还要复杂。当代理是原型时,仅当默认操作通常继续到原型时才会调用代理trap,这会将代理的功能限制为原型。如下:

var target = {};
var newTarget = Object.create(new Proxy(target, {

    // never called
    defineProperty(trapTarget, name, descriptor) {

        // would cause an error if called
        return false;
    }
}));

Object.defineProperty(newTarget, "name", {
    value: "newTarget"
});

console.log(newTarget.name);                    // "newTarget"
console.log(newTarget.hasOwnProperty("name"));  // true

newTarget代理是作为一个原型对象被创建。现在,只有在newTarget上的操作并将操作传递到目标(target)上时,这样才会调用代理trap.

definePropertynewTarget的基础上创建了自己的属性name,在对象上定义属性,不会作用到原型, 可以看下原型的影子方法,并且不会调用代理的defineProperty trap,只会把这个name属性添加到自己的属性里。

虽然代理在用作原型时受到严重限制,但仍有一些陷阱仍然有用。

Using the get Trap on a Prototype

我们知道原型链的查找是现在自己的属性里查找,如果找不到会遍历原型链。因此,只需要给代理设置一个get trap,当查找的属性不存在的时候,就会触发原型上的trap

var target = {};
var thing = Object.create(new Proxy(target, {
    get(trapTarget, key, receiver) {
        throw new ReferenceError(`${key} doesn't exist`);
    }
}));

thing.name = "thing";

console.log(thing.name);        // "thing"

// throw an error
var unknown = thing.unknown;

可以发现,使用代理作为原型创建thing对象。当调用不存在的时候,会抛出错误,如果存在,便不会遍历到原型,所以不会出错。

在这个例子中,要理解trapTargetreceiver是不同的对象。当代理当做原型使用时,trapTarget是原型对象本身,receiver是实例对象。在上例中,trapTarget等同于target, receiver等同于thing

Using the set Trap on a Prototype

这个比较麻烦,如果赋值操作继续到原型,触发这个trap,他会根据参数情况确定是在原型上或者是在当前实例上创建属性,他的默认情况就和我们上面说的影子方法一样。这里可能有些绕,可以看看下面这个例子:

var target = {};
var thing = Object.create(new Proxy(target, {
    set(trapTarget, key, value, receiver) {
        return Reflect.set(trapTarget, key, value, receiver);
    }
}));

console.log(thing.hasOwnProperty("name"));      // false

// triggers the `set` proxy trap
thing.name = "thing";

console.log(thing.name);                        // "thing"
console.log(thing.hasOwnProperty("name"));      // true

// does not trigger the `set` proxy trap
thing.name = "boo";

console.log(thing.name);                        // "boo"

在这个例子中,target没有自己的属性。 thing对象有一个代理作为其原型,它定义了一个set trap来捕获任何新属性的创建。当thing.name被赋值为“thing”作为其值时,将调用代理set trap,因为thing没有自己的name属性。在这个set trap里,trapTarget等于targetreceiver等于thing。该操作最终在thing上创建一个新属性,幸运的是,如果你将receiver作为第四个参数传入,Reflect.set会为你实现这个默认行为。

如果不传递这个第四个receiver参数呢,那么就会在原型对象上(target)创建属性, 不会在实例上创建属性,那么就导致每次set都去原型操作;如果传递之后,那么在设置过一次就不会去再次触发原型上的set trap.

Proxies as Prototypes on Classes

类不可以直接修改原型做代理,因为prototype属性是不可写的。

'use strict'
class X {}
X.prototype = new Proxy({}, {
    get(trapTarget, key, receiver){
        console.log('class prototype proxy')
    }
})
// Cannot assign to read only property 'prototype' of function 'class X {}'

但是,可以创建一个通过使用继承将代理作为其原型的类。首先,需要使用构造函数创建ES5样式类型定义。然后,用原型覆盖为代理。

function NoSuchProperty() {
    // empty
}

NoSuchProperty.prototype = new Proxy({}, {
    get(trapTarget, key, receiver) {
        throw new ReferenceError(`${key} doesn't exist`);
    }
});

var thing = new NoSuchProperty();

// throws error due to `get` proxy trap
var result = thing.name;

函数的prototype属性没有限制,因此可以使用代理覆盖它。

接下来就是创建一个类去继承这个函数。

function NoSuchProperty() {
    // empty
}

NoSuchProperty.prototype = new Proxy({}, {
    get(trapTarget, key, receiver) {
        throw new ReferenceError(`${key} doesn't exist`);
    }
});

class Square extends NoSuchProperty {
    constructor(length, width) {
        super();
        this.length = length;
        this.width = width;
    }
}

var shape = new Square(2, 6);

var area1 = shape.length * shape.width;
console.log(area1);                         // 12

// throws an error because "anotherWidth" doesn't exist
var area2 = shape.length * shape.anotherWidth;

这样,就很好的在原型上使用了代理,一个折中的法子来实现。

我们来该写下,这样可能会更直观:

function NoSuchProperty() {
    // empty
}

// store a reference to the proxy that will be the prototype
var proxy = new Proxy({}, {
    get(trapTarget, key, receiver) {
        throw new ReferenceError(`${key} doesn't exist`);
    }
});

NoSuchProperty.prototype = proxy;

class Square extends NoSuchProperty {
    constructor(length, width) {
        super();
        this.length = length;
        this.width = width;
    }
}

var shape = new Square(2, 6);

var shapeProto = Object.getPrototypeOf(shape);

console.log(shapeProto === proxy);                  // false

var secondLevelProto = Object.getPrototypeOf(shapeProto);

console.log(secondLevelProto === proxy);            // true

这里,把代理存在变量中,更加直观。在这里shape的原型是Square.prototype,不是proxy。但是Square.prototype的原型是代理,因为他继承自NoSuchProperty

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容

  • 原创文章&经验总结&从校招到A厂一路阳光一路沧桑 详情请戳www.codercc.com 主要知识点:代理和反射的...
    你听___阅读 1,957评论 0 1
  • 本人自学es6已经有一段时间了,只觉得有些时候很是枯燥无味, 时而又觉得在以后的职业生涯中会很有用,因为es6的很...
    可乐_37d3阅读 1,523评论 0 0
  • 大婶我一把年纪,还喜欢泡韩剧,帅锅美女扎堆,少女心碎一地。言归正传,韩剧里的美食,也是让人口水流一地啊!没钱去飞去...
    Dy倒影阅读 311评论 1 3
  • 因为最近越发自己的孤独,自己的一个人,甚至于想找个人打羽毛球也找不到。生活里好像剩下了自己一个人。虽然住在宿舍了,...
    自然清醒爱自己阅读 161评论 0 0
  • 我喜欢在大雨里奔跑 我喜欢光脚踩在沙滩上 我喜欢把书盖在脸上睡觉 我喜欢在纸上 反复写一个人的名字 我喜欢他带我穿...
    妖妖z阅读 328评论 16 14