8

我正在尝试在函数代理上调用 .toString() 。

简单地创建一个函数代理并调用 toString 会导致“TypeError:Function.prototype.toString is not generic”,设置 toString 以返回原始导致“RangeError:超出最大调用堆栈大小”的源,但是为 toString 创建了一个 get 陷阱作品。

为什么简单地设置 toString 函数不起作用,但做一个 get 陷阱呢?

function wrap(source) {
 return(new Proxy(source, {}))
}
wrap(function() { }).toString()

function wrap(source) {
 let proxy = new Proxy(source, {})
 proxy.toString = function() {
  return(source.toString())
 }
 return(proxy)
}
wrap(function() { }).toString()

function wrap(source) {
 return(new Proxy(source, {
  get(target, key) {
   if(key == "toString") {
    return(function() {
     return(source.toString())
    })
   } else {
    return(Reflect.get(source, key))
} } })) }
wrap(function() { }).toString()

4

2 回答 2

6

我遇到了同样的问题。我终于发现这是一个问题this。向您的处理程序添加一个get陷阱,将代理对象绑定到this代理属性上(如果它是 a function),它似乎可以正常工作:

function wrap(source) {
    return new Proxy(source, {
        get: function (target, name) {
            const property = target[name];
            return (typeof property === 'function') 
                ? property.bind(target)
                : property;
        }
    });
}

console.log(wrap(function () {}).toString());

于 2017-02-25T21:31:59.100 回答
1

TypeError:Function.prototype.toString 不是通用的

似乎Function.prototype.toString不应该被调用Proxy

proxy.toString = function() {

这个对代理的分配被传递给source对象,因为您没有分配陷阱。如果你检查source.hasOwnProperty('toString')你会得到true. 添加get陷阱时,您不会更改toString方法,也不会将其添加到source对象中,因此可以正常工作。

另一个可能的解决方案是

function wrap(source) {
  let proxy = new Proxy(source, {})
  proxy.toString = Function.prototype.toString.bind(source)
  return proxy
}
于 2016-07-11T22:27:38.407 回答