9

我正在尝试找到一种获取构造函数名称的通用方法。我的目标是为 KnockoutJS 创建一个约定而不是配置框架

我的想法是遍历窗口中的所有对象,当我找到我正在寻找的构造函数时,我可以使用索引来获取构造函数的名称

到目前为止的代码

(function() {
    constructors = {};
    window.findConstructorName = function(instance) {
        var constructor = instance.constructor;
        var name = constructors[constructor];
        if(name !== undefined) {
            return name;
        }

        var traversed = [];
        var nestedFind = function(root) {       
            if(typeof root == "function" || traversed[root]) {
                return       
            }

            traversed[root] = true;
            for(var index in root) {               
                if(root[index] == constructor) {
                    return index;
                }


                var found = nestedFind(root[index]);  
                if(found !== undefined) {
                    return found;
                }
            }
        }

        name = nestedFind(window);
        constructors[constructor] = name;
        return name;
    }
})();

var MyApp = {};
MyApp.Foo = function() {
};

var instance = new MyApp.Foo();
console.log(findConstructorName(instance));     

问题是我得到了一个Permission denied to access property 'toString'异常,我什至不能尝试 catch 所以看看是哪个对象导致了问题

小提琴http://jsfiddle.net/4ZwaV/

这个小提琴的最终版本 http://jsfiddle.net/2Uvd5/8/

在此处查看我的公约配置插件的雏形 https://github.com/AndersMalmgren/Knockout.BindingConventions

4

1 回答 1

5
  • 编辑2:

JSFiddle

这解决了所有问题,除了一件事:var MyApp = {}; 将其添加到window-object. 将其更改为window.MyApp = {};使其完全正常工作(即使在 IFrame 中)。


  • 编辑1:

JSFiddle

通过设置键名添加到数组中需要键名是字符串,以便 Javascript 自动调用。toString()在您建议的键名上,这对于某些对象将失败。而是使用 .push() 将任何类型的元素添加到数组中,然后.indexOf()检查它是否已经存在。

请注意,jsFiddle 仍然会因为放置在iframe. 在新选项卡中打开它可以解决这个问题。


我之前的回答(当我试图在你的 jsFiddle 中验证它时证明是无效的):

您需要检查构造函数是否是一个精确的对象。如果它在它上面调用 .toString() 将导致一个安全异常,我发现它有点难以调试。这是我用来在我使用的 var-dumper 中获取对象类型的函数。

function GetTypeOfObject(obj) {
    if (obj.constructor === window.Object)
        return '[object]';
    else
        return obj.constructor.toString();
}
于 2012-12-13T17:38:19.507 回答