将JavaScript构造函数作为函数调用(没有新的?)是否存在问题

Are there problems with calling JavaScript constructors as functions(without new?)

本文关键字:是否 问题 存在 构造函数 JavaScript 函数调用      更新时间:2023-09-26

最近,我养成了调用RegExp、String、Number、Object、TypeError等不带"new"的习惯。

例如:

throw (TypeError("Error"));
var regex = RegExp('^word$');

我知道,当需要"this"上下文时,这是不好的,因为如果没有"new","this"会对你的全局范围造成严重破坏,除非你用"use strict"包装你的代码,在这种情况下,它会引发一个错误,即你正试图变异为"undefined"。(我不确定这是否适用于非常旧的浏览器(。

例如:

var constructor = function() {
    // 'use strict'; /* uncomment this line to avoid the behavior and be warned */
    this.state = 'working as intended';
};
var foo = constructor();
console.log(foo.state); // undefined
console.log(window.state); // we just polluted our global scope.

var constructor = function() {
    this.state = 'working as intended';
};
var foo = new constructor;
console.log(foo.state); // "working as intended"
console.log(window.state); // we are clean.

但在像上面这样的情况下,这样做可以吗?或者如果我养成了这样做的习惯,我会遇到问题吗?

提前谢谢。

请注意,结果可能会有所不同。

例如,Number构造函数创建Number对象,但当作为函数调用时,它只对基元Number进行类型强制。

new Number(123); // Number { 123 }
Number(123); // 123

但是的,在很多情况下,是否使用new并不重要。它们的存在是因为向后兼容性,但最近引入的构造函数(如SetMap(确实需要new

通常,当您想要创建一个新对象时,我建议您使用new。然后,

  • 如果只想执行类型强制,则必须调用不带newBooleanNumberString
  • 出于同样的原因,如果您想强制为Object类型,我不会使用new,但这并不重要
  • 如果要创建一个没有文字语法的基元,则必须调用不带newSymbol
  • 如果要创建基元值的对象包装器,则必须使用new调用BooleanNumberStringSymbol
  • 如果您想实例化一个旧的构造函数,如ArrayObjectRegExpError等,我会使用new,但这无关紧要
  • 如果要实例化最近引入的构造函数,如SetMapWeakSetWeakMap、类型化数组等,则必须使用new调用它

对于那些无关紧要的老构造函数,如果省略它,它们就好像用new来调用自己一样。例如,对于RegExp

RegExp作为函数而不是构造函数调用时创建并初始化一个新的RegExp对象。因此函数调用RegExp(…)等效于对象创建表达式具有相同自变量的new RegExp(…)

如另一个答案所述,一些内置构造函数被编写为能够作为函数调用。所以,只要你没有污染你的全局命名空间,我相信你应该对你列出的例子很满意。

话虽如此,在大多数情况下,我不会习惯这种习惯;new运算符增加了比简洁更重要的代码清晰度。在没有new运算符的情况下使用本机构造函数的能力在构造函数之间的应用也不一致。