newFunction()语法
学习《ECMAScript6⼊门》中的模板字符串的案例中看见了new Function()创建函数的语法:
let str = 'return ' + '`Hello ${name}!`';
let func = new Function('name', str);
func('Jack') // "Hello Jack!"
上⾯的代码传⼊name参数和字符串函数体,字符串函数体由模板字符串提供,⾮常简洁⽅便。
为理解new Function,于是到了下⾯这篇⽂章:
下⽂翻译⾃:
这⾥有⼀个很少被⽤到的新建函数的⽅法,但是有时候不得不使⽤它。
语法
新建函数的语法:
let func = new Function ([arg1[, arg2[, ...argN]],] functionBody)
换句话说,函数的参数(或更确切地说,各参数的名称)⾸先出现,⽽函数体在最后。所有参数都写成字符串形式。
通过查看⽰例,可以更容易理解。这是⼀个有两个参数的函数:
let sum = new Function('a', 'b', 'return a + b');
alert( sum(1, 2) ); // 3
如果所要新建的函数没有参数,那么new Function()只有⼀个函数体参数:
let sayHi = new Function('alert("Hello")');
sayHi(); // Hello
这个⽅式与其他⽅式最主要的不同点在于,函数是由在运⾏时传⼊的字符串创建的。
之前的所有声明都要求程序员在脚本中编写功能代码。
但new Function允许将任何字符串转换为函数。例如,我们可以从服务器接收新函数然后执⾏它:
let str = ... receive the code from a server dynamically ...
let func = new Function(str);
func();
它在⾮常特殊的情况下使⽤,例如当我们从服务器接收代码时,或者使⽤模板动态编译函数。对此的需求通常出现在开发的进阶阶段。闭包(Closure)
通常,函数将它所创建的位置记录在特殊属性[[Environment]]中。它引⽤了创建地点的词法环境。
但是当使⽤new Function()创建函数时,其[[Environment]]不是引⽤当前的词法环境,⽽是引⽤全局环境。
function getFunc() {
let value = "test";
let func = new Function('alert(value)');能够删除字符串中空格的函数是
return func;
}
getFunc()(); // error: value is not defined
与常规⽅法⽐较:
function getFunc() {
let value = "test";
let func = function() { alert(value); };
return func;
}
getFunc()(); // "test", 来⾃getFunc的词法环境
这个特殊的new Function表⾯看起来很奇怪,但在实践中显得⾮常有⽤。
想象⼀下,我们必须从字符串创建⼀个函数。在编写脚本时不知道该函数的代码(这就是我们不使⽤常规函数的原因),但在执⾏过程中将会知道。我们可能会从服务器或其他来源收到它。
我们的新函数需要与主脚本进⾏交互。
也许我们希望它能够访问外部的局部变量?
问题是在JavaScript发布到⽣产之前,它是使⽤minifier压缩的——⼀个通过删除额外的注释,空格来缩⼩代码的特殊程序,⽽且 - 重要的是,会将局部变量重命名为较短的变量。
例如,如果⼀个函数中有let userName,那么minifier会替换它为let a(或者如果a被占⽤,则⽤另⼀个字母替换),这个过程会在任何地⽅进⾏。这通常是⼀件安全的事情,因为被替换的变量是局部的,函数外部没有任何东西可以访问它。并且在函数内部,minifier取代了它的每⼀个提及。Minifiers很聪明,他们会分析代码结构,所以他们不会破坏任何东西。他们不是只会愚蠢地发现和替换。
但是,如果new Function可以访问外部变量,那么它将⽆法到userName,因为userName在代码缩⼩后才作为字符串传⼊。
所以,即使我们可以在new Function中访问外部词汇环境,我们也会遇到minifiers的问题。
⽽这时,new Function的“特⾊”可以让我们免于犯错。
它强制执⾏更好的代码。如果我们需要将某些东西传递给由new Function创建的函数,我们应该将它作为参数显式传递(可以避免直接读取外部变量时产⽣的问题)。
我们的“sum”函数实际上是这样正确使⽤的:
let sum = new Function('a', 'b', 'return a + b');
let a = 1, b = 2;
// 外部变量作为参数传⼊
alert( sum(a, b) ); // 3
总结
语法:
let func = new Function(arg1, arg2, ..., body);
历史原因,参数也可以以逗号分隔的列表的形式给出。
这三个意思相同:
new Function('a', 'b', 'return a + b'); // 基础语法
new Function('a,b', 'return a + b'); // 逗号分隔
new Function('a , b', 'return a + b'); // 逗号加空格分隔
使⽤new Function创建的函数,其[[Environment]]引⽤全局词法环境,⽽不是包含该函数的外部词法环境。因此,他们不能使⽤外层的变量。但这确实很好,因为它可以使我们免于错误。明确地传递参数在架构上是⼀种更好的⽅法,并且不会在使⽤minifiers时不会产⽣问题。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论