javascript - 将 .apply() 与 'new' 运算符一起使用。这可能吗?

标签 javascript oop class inheritance constructor

在 JavaScript 中,我想创建一个对象实例(通过 new 运算符),但将任意数量的参数传递给构造函数。这可能吗?

我想做的是这样的(但下面的代码不起作用):

function Something(){
    // init stuff
}
function createSomething(){
    return new Something.apply(null, arguments);
}
var s = createSomething(a,b,c); // 's' is an instance of Something

答案

从此处的响应可以清楚地看出,没有使用 new 运算符调用 .apply() 的内置方法。然而,人们针对这个问题提出了许多非常有趣的解决方案。

我的首选解决方案是 this one from Matthew Crumley (我修改了它以传递 arguments 属性):

var createSomething = (function() {
    function F(args) {
        return Something.apply(this, args);
    }
    F.prototype = Something.prototype;

    return function() {
        return new F(arguments);
    }
})();

最佳答案

使用 ECMAScript5 的 Function.prototype.bind事情变得很干净:

function newCall(Cls) {
    return new (Function.prototype.bind.apply(Cls, arguments));
    // or even
    // return new (Cls.bind.apply(Cls, arguments));
    // if you know that Cls.bind has not been overwritten
}

可以按如下方式使用:

var s = newCall(Something, a, b, c);

甚至直接:

var s = new (Function.prototype.bind.call(Something, null, a, b, c));

var s = new (Function.prototype.bind.apply(Something, [null, a, b, c]));

这个和 eval-based solution是唯一始终有效的方法,即使使用像 Date 这样的特殊构造函数也是如此:

var date = newCall(Date, 2012, 1);
console.log(date instanceof Date); // true

编辑

一点解释: 我们需要在接受有限数量参数的函数上运行 newbind 方法允许我们这样做:

var f = Cls.bind(anything, arg1, arg2, ...);
result = new f();

anything 参数无关紧要,因为 new 关键字会重置 f 的上下文。但是,出于句法原因,它是必需的。现在,对于 bind 调用:我们需要传递可变数量的参数,所以这样做:

var f = Cls.bind.apply(Cls, [anything, arg1, arg2, ...]);
result = new f();

让我们把它包装在一个函数中。 Cls 作为参数 0 传递,因此它将是我们的任何内容

function newCall(Cls /*, arg1, arg2, ... */) {
    var f = Cls.bind.apply(Cls, arguments);
    return new f();
}

实际上,根本不需要临时的f变量:

function newCall(Cls /*, arg1, arg2, ... */) {
    return new (Cls.bind.apply(Cls, arguments))();
}

最后,我们应该确保 bind 确实是我们所需要的。 (Cls.bind 可能已被覆盖)。所以将其替换为Function.prototype.bind,最终结果如上。

关于javascript - 将 .apply() 与 'new' 运算符一起使用。这可能吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20789044/

相关文章:

javascript - 获取javascript类中静态方法的值

javascript - 如何使用 ng-bootstrap (Angular 2) 显示简单模式

javascript - 我该如何做才能使路径在内容下方延伸?

c++ - 我们可以像 C 中的结构一样使用 C++ 中的类吗?

JavaScript setInterval 范围问题 : an alternative to bind

html - 从 font-awesome v4 到 v5

javascript - 使用属性等于选择器时转义元素名称中的方括号

javascript - 使用 jodit 编辑器在 textarea 中写入或删除字符串

c# - Fluent APIs - 返回这个还是新的?

c++ - OOP - 这是不好的做法吗?