スコープのネストはスコープチェーンを形成し、関数のネストはクロージャを形成します。クロージャとスコープ チェーンは、JavaScript を他の言語と区別する重要な機能の 1 つです。
スコープ
JavaScript には、関数スコープとグローバル スコープの 2 種類のスコープがあります。
関数内で宣言された変数と関数のパラメーターは同じスコープ、つまり関数スコープを共有します。関数スコープの簡単な例:
C などの他のブロックスコープ言語とは異なり、これは常に 2 を返します。
ブラウザーのグローバル スコープは、ウィンドウ オブジェクトとして理解できます (Node.js はグローバルです):
変数 bar と関数 foo は両方ともグローバル スコープに属し、両方とも window の属性です。
スコープ チェーン
JavaScript で変数にアクセスする場合、ローカル変数とパラメーターから開始され、グローバル スコープに到達するまでスコープを段階的に移動します。
クロージャ
ある関数で別の関数を定義することを関数の入れ子と呼びます。関数をネストするとクロージャが形成されます。
クロージャとスコープ チェーンは相互に補完し、関数をネストすると、チェーン関係に複数のスコープが作成されるだけでなく、クロージャも形成されます。
外部関数は組み込み関数のパラメータや変数にアクセスできません
ただし、組み込み関数は外部関数のパラメータや変数にアクセスできます言い換えると、組み込み関数には、外部関数
前述のスコープ チェーンの例を見てみましょう。今回はクロージャの観点から理解します。
最も内側の関数は、内部および外部で定義されたすべての変数にアクセスできます。最後から 2 番目の層の関数は、最内層の変数にアクセスできません。同時に、最内層のスコープ = 3 の代入操作は、同じ名前の外部変数には影響しません。
クロージャを別の角度から理解してみましょう:
外部関数が呼び出されるたびに、埋め込み関数が一度作成されます。
作成されると、外部関数のスコープ (ローカル変数、パラメータなどのコンテキストを含む) が各埋め込み関数オブジェクトになります。外部関数が実行を完了して終了した後でも、内部状態の一部
次の例を参照してください:
予想される "1" と "2" の代わりに "2" が 2 回取得されます。これは、リスト内の 2 つの関数によってアクセスされる変数 i が、上位スコープ内の同じ変数であるためです。
この問題を解決するためにクロージャを使用するようにコードを変更しましょう:
外側の「即時実行関数」は、関数内にパラメーター j の形式で存在するパラメーター変数 i を受け取り、返された内部関数内の名前 j と同じ参照を指します。外側の関数が実行されて終了すると、パラメーター j (その値はこの時点での i の現在の値) が内側の関数の状態の一部となり、保存されます。