Javascript のスコープとクロージャの詳細

Javascript のスコープとクロージャの詳細

1. 範囲

簡単に言えば、スコープとは、変数が定義されているプログラム内の領域を指し、現在実行されているコードの変数へのアクセス権を決定します。

ES5 では、一般的にスコープ タイプは次の 2 つだけです。

  • グローバルスコープ:グローバルスコープはプログラムの最も外側のスコープであり、常に存在します。
  • 関数スコープ:関数スコープは、関数が定義され、親関数スコープまたはグローバル スコープに含まれている場合にのみ作成されます。

概念について説明した後、次のコードを見てみましょう。

変数a = 100
関数テスト(){
    var b = a * 2
    変数a = 200
    変数 c = a/2
    コンソールログ(b)
    コンソール.log(c)
}
test() // ここで何が印刷されるでしょうか?

分析:

  • まず、このコードはグローバルスコープと関数スコープを形成します
  • グローバルスコープに100の値を持つ変数aがある
  • ローカル変数 b、a、c はテスト関数スコープ内で定義されます。
  • ここでは変数の昇格が行われています。変数の昇格はまず関数スコープ内で実行されます: var b; var a; var c;
  • 次に、b に値を割り当てます。この時点では、a には値が割り当てられていないため、a の値は未定義です。次に、a*2 が加算されるため、b は NaN になります。
  • 次に、a に値 200 を割り当て、c に値 a/2 (つまり 100) を割り当てます。

つまりNaN、100が出力される。

ES6では、新しいブロックスコープが追加されました

簡単に言えば、中括弧{...}内の領域がブロックレベル スコープですが、 Javascriptブロックレベル スコープをネイティブにサポートしていません。ブロックレベル スコープを作成するには、 ES6で提案されているletconstを使用する必要があります。

// ES5
真の場合{
  var name = '南九'
}
console.log(name) // ナンジュウ // ES6
真の場合{
  年齢を18とする
}
console.log(age) // エラーが報告されます

2. スコープチェーン

実行可能コード内の変数にアクセスする場合、まずその変数が現在のスコープ内に存在するかどうかを確認します。存在する場合は、すぐに戻ります。存在しない場合は、親スコープ内を検索し、グローバル スコープが見つかるまで検索を続けます。このスコープ ネスト メカニズムを作用域鏈

3. 語彙の範囲

詞法作用域は、スコープの実用的なモデルです。レキシカル スコープは、 JavaScriptで使用されるスコープの一種です。レキシカル スコープは、靜態作用域とも呼ばれます。

いわゆるレキシカルスコープは、コードを書くときに変数とスコープをどこに記述するかによって決まります。つまり、レキシカルスコープは静的スコープであり、コードを書くときに決定されます。関数のスコープは、実際に呼び出される場所ではなく、宣言される場所によって決まります。

MDN ではクロージャを次のように定義しています。

関数は、その周囲 (語彙環境) への参照とバンドルされます (または、関数は参照に囲まれています)。このような組み合わせは、 closureと呼ばれます。

つまり、クロージャを使用すると、内部関数内から外部関数のスコープにアクセスできるようになります。 JavaScriptでは、関数が作成されるたびに、関数の作成と同時にクロージャも作成されます。

次のように結論付けることができます。

閉包= 函數+ 外層作用域

まずはコードを見てみましょう:

var name = 'フロントエンドNanjiu'

関数 say() {
  console.log(名前)
}
言う()

分析: say関数は外側のスコープ内の変数 a にアクセスできるので、これはクロージャを形成しませんか?

「Javascript Definitive Guide」という本に次のような一文があります。厳密に言えば、すべてのJavaScript関数はクロージャです。

しかし、これは単なる理論的な閉鎖であり、私たちが通常使用するものとは異なります。上記の例は単なる単純なクロージャです。

ECMAScript ではクロージャを次のように定義します。

  • 理論上、すべての関数はクロージャです。作成されたときに上位コンテキストのデータがすでに保存されているからです。
  • 実際には、クロージャは次の 2 つの条件を満たす必要があります。1. 外側のスコープの変数がコード内で参照されていること。2. クロージャが作成されたコンテキストが破棄されても、クロージャは存在し続けていることです。

JavaScript Definitive Guide の別のコードを見てみましょう。

スコープを「グローバルスコープ」にする
関数チェックスコープ(){
  スコープを 'ローカルスコープ' にします
  関数f(){
    戻りスコープ
  }
  戻り値 f
}

s = checkscope() とします。   
s() // これは何を返しますか?

多くの学生はそれがglobal scopeであると考えるかもしれませんが、本当にそうでしょうか? その実行プロセスを見てみましょう。

  • まず、グローバルコードを実行し、グローバル実行コンテキストを作成し、グローバル変数のscopeを定義して値を割り当てます。
  • checkscope関数を宣言し、関数の実行コンテキストを作成し、ローカル変数scopeを作成して値を割り当てます。
  • f関数を宣言し、関数の実行コンテキストを作成します。
  • checkscope関数を実行し、f関数を返して変数sに代入します。
  • s 関数を実行することは、f 関数を実行することと同じです。ここで返されるscopelocal scopeです。なぜlocal scopeなのかについては、語彙についてお話ししました。

スコープの基本ルール: JavaScript関数は、定義されているスコープを使用して実行されます。 f関数が定義されているスコープでは、変数scopeの値はlocal scopeである

5. 閉鎖の適用

クロージャは主に内部変数を維持するために使用されます。

6. クロージャの欠陥

  • クロージャの存在により変数がメモリ内に存在する可能性があるため、不適切な使用はメモリリークを引き起こす可能性があります。
  • メモリリークによりアプリがフリーズしたりクラッシュしたりする可能性がある

7. 閉会面接でよくある質問

var arr = []
(var i=0;i<3;i++){
    arr[i] = 関数(){
        コンソールログ(i)
    } 
}
arr[0]() // 3
arr[1]() // 3
arr[2]() // 3
// ここで、実行中に i は 3 になりました

// クロージャを使用して解決する var arr = []
(var i=0;i<3;i++){
    arr[i] = (関数(i){
        関数()を返す{
            コンソールログ(i)
        } 
    })(私)
    
}
arr[0]() // 0
arr[1]() // 1
arr[2]() // 2

Javascript のスコープとクロージャの詳細に関するこの記事はこれで終わりです。Javascript のスコープとクロージャの詳細については、123WORDPRESS.COM の以前の記事を検索するか、以下の関連記事を引き続き参照してください。今後とも 123WORDPRESS.COM をよろしくお願いいたします。

以下もご興味があるかもしれません:
  • JS の難しさ 同期と非同期、スコープとクロージャ、プロトタイプとプロトタイプ チェーンの詳細な説明
  • JavaScript スコープ、スコープ チェーン、クロージャの使用方法の詳細な説明
  • JavaScript スコープクローズの詳細な説明
  • JS スコープとクロージャの詳細な理解
  • JavaScript 関数の使用方法の詳細な説明 [関数の定義、パラメータ、バインディング、スコープ、クロージャなど]
  • JS ページはセッション値、スコープ、およびクローズ スタディ ノートを取得します
  • JavaScriptはクロージャを使用してブロックレベルのスコープ操作をシミュレートします
  • JavaScript のスコープとクロージャ

<<:  CSSカウンター関連属性の学習の詳細な説明

>>:  Docker stopはすべてのコンテナを停止/削除します

推薦する

Linux リモート管理と sshd サービス検証の知識ポイントの詳細な説明

1. SSHリモート管理SSH の定義SSH (Secure Shell) は、主にキャラクタ イン...

iframe が HTML 内のページにジャンプするのを防ぎ、iframe を使用して WeChat Web バージョンをページに埋め込む方法

私は、WinForm と HTML5 を組み合わせた小さなものを作りたいだけなのですが、突然、そこに...

MySQLのkillがスレッドをkillできない理由

目次背景問題の説明原因分析シミュレーションする総括する背景日常の使用において、MySQL で個別また...

MySQL がエラーを報告: ファイルが見つかりません: './mysql/plugin.frm' 解決策

問題を見つける最近、仕事中に問題が見つかりました。問題は、MySQL ディスクがいっぱいだったことで...

6秒でMySQLに100万件のレコードを挿入する方法を教えます

1. アイデアMySQL に 1,000,000 件のレコードを挿入するのにたった 6 秒しかかかり...

一定時間後にNavicatがデータベースから自動的に切断される問題の解決方法

これは、データベース サーバーが、接続が多すぎるのを避けるために、一定時間非アクティブな状態が続くと...

Nginx の負荷分散アルゴリズムとフェイルオーバー分析

概要Nginx ロード バランシングは、アップストリーム サーバー (実際のビジネス ロジックによっ...

MySQL 全文インデックスガイド

全文インデックスには特別なクエリ構文が必要です。全文検索はインデックスの有無にかかわらず実行できます...

MySQLカーソルの使い方と機能の詳細な説明

[mysqlカーソルの使い方と機能]例:現在、テーブル A、B、C の 3 つのテーブルがあります。...

ミニプログラムカスタムタブバーコンポーネントのカプセル化

この記事の例では、ミニプログラムのカスタムタブバーコンポーネントをカプセル化するための具体的なコード...

MySQL バッチ挿入ループの詳細なサンプルコード

背景数日前、MySql でページングを行っていたときに、ページングに制限 0,10 を使用するとデー...

WeChatミニプログラムが星評価を実装

この記事では、WeChatアプレットで星評価を実装するための具体的なコードを参考までに紹介します。具...

CentOS 7 で RPM パッケージを使用して MySQL 5.7.9 をインストールするチュートリアル

MySQL 5.7.9 のインストールチュートリアルを録画してみんなと共有しましょう環境の紹介:オペ...

MySQL オンラインリカバリ UNDO テーブルスペース 実戦記録

1 MySQL5.6 1.1 関連パラメータMySQL 5.6 では、innodb_undo_dir...

CSS の ::before と ::after 疑似要素について知らないこと

CSS には、一般的には使用されない 2 つの疑似クラス、before と :after があります...