2022年1月2日

"new Function" 構文

関数を作るもう1つの方法があります。ほとんど使われませんが、代替手段がないことがたまにあります。

構文

関数を作る構文です:

let func = new Function ([arg1[, arg2[, ...argN]],] functionBody)

引数 arg2...argN と指定された functionBody で関数が作成されます。

例を見ると理解し易いです。以下は2つの引数を持つ関数です:

let sum = new Function('a', 'b', 'return a + b');

alert( sum(1, 2) ); // 3

引数がない場合は関数本体だけを指定します:

let sayHi = new Function('alert("Hello")');

sayHi(); // Hello

これまで見てきたような他の方法との大きな違いは、関数は文字通り文字列から作られ、実行時に渡されるということです。

これまでのすべての宣言は、プログラマーがスクリプトに関数コードを書く必要がありました。

しかし、new Function は任意の文字列を関数にすることができます。例えば、サーバから新しい関数を受け取りそれを実行することができます:

let str = ... サーバから動的にコードを受け取る ...

let func = new Function(str);
func();

これは 複雑なWebアプリケーションでサーバからコードを受け取ったり、テンプレートから動的に関数をコンパイルするような、非常に特定のケースで使われます。

クロージャ

通常、関数は特別なプロパティ [[Environment]] でどこで生成されたかを覚えています。それは作成された場所からレキシカル環境を参照します(変数スコープ、クロージャ の章で説明しました)。

しかし、new Function を使用して作られた関数の場合、その [[Environment]] は現在のレキシカル環境ではなく、グローバルのレキシカル環境を参照します。

そのため、このような関数は外部変数へのアクセスは持たず、グローバル変数のみとなります。

function getFunc() {
  let value = "test";

  let func = new Function('alert(value)');

  return func;
}

getFunc()(); // error: value は未定義

通常の振る舞いとの比較です:

function getFunc() {
  let value = "test";

  let func = function() { alert(value); };

  return func;
}

getFunc()(); // "test", getFunc のレキシカル環境から

この new Function の特殊な機能は奇妙に見えますが、実践では非常に役立ちます。

本当に文字列から関数を作る必要がある場合をイメージしてください。その関数のコードはスクリプト生成時には知られていません(そういう訳で通常の関数を使うことができません)が、実行中に認識されます。我々はサーバや別のソースからそれを受け取ることができます。

新しい関数はメインのスクリプトと相互にやり取りが必要です。

外部変数へアクセスしたいときはどうするのでしょうか?

問題はJavaScriptが本番環境に公開される前に、minifier(余分なコメントやスペースなどを削除することでコード小さくする特別なプログラム)を使用して圧縮されており、重要なことはローカル変数をより短いものにリネームします。

例えば、もし関数が let userName を持っていたとき、minifier はそれを let a (または既に使われていれば別の文字) に置き換え、その変数を使用します。変数はローカルであり関数の外部からアクセスすることはできないため、それは通常安全です。また、関数の内側では minifier はそれに関する全ての箇所を置き換えます。Minifiers は賢いので、単なる検索と置換ではなく、コード構造を分析するので問題ありません。

そのため、new Function が外部変数へアクセスする場合、リネームされた userName を見つけることはできません。

new Function が外部変数へアクセスする場合、minifiers で問題になります。

加えて、このようなコードはアーキテクチャ的に良くなく、エラーになりやすいです。

new Function として作成された関数に何かを渡すには、その引数を使用しなければなりません。

サマリ

構文:

let func = new Function ([arg1, arg2, ...argN], functionBody);

歴史的な理由から、引数はカンマ区切りのリストで与えられます。

これらの3つの意味は同じです:

new Function('a', 'b', ' return a + b; '); // 基本構文
new Function('a,b', ' return a + b; '); // カンマ区切り
new Function('a , b', ' return a + b; '); // スペースありのカンマ区切り

new Function で作られた関数は グローバルレキシカル環境を参照する [[Environment]] を持っており、外部のレキシカル環境ではありません。従って、それらは外部の変数を使うことができません。ですが、それは良いことです。明示的なパラメータ渡しはアーキテクチャ的にはよりよい方法であり、minifierの問題も起きません。

チュートリアルマップ

コメント

コメントをする前に読んでください…
  • 自由に記事への追加や質問を投稿をしたり、それらに回答してください。
  • 数語のコードを挿入するには、<code> タグを使ってください。複数行の場合は <pre> を、10行を超える場合にはサンドボックスを使ってください(plnkr, JSBin, codepen…)。
  • 記事の中で理解できないことがあれば、詳しく説明してください。