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はすべてのコンテナを停止/削除します

推薦する

MySql 共通クエリコマンド操作リスト

MYSQL でよく使用されるクエリ コマンド: mysql> select version()...

JavaScript ファクトリーパターンの説明

目次シンプルファクトリーファクトリーメソッド安全な工場方法アブストラクトファクトリー要約するシンプル...

Centos 7 sshd の変更 | ルートログインの禁止と sshd ポートスクリプトの定義

1. 新しいユーザーwwweee000を作成します [root@localhost ~]# user...

mysql ステートメントを使用してユーザー権限を照会するプロセスの詳細な説明

MySQL では、ユーザーに付与された権限をどのように確認しますか? ユーザーに付与される権限は、グ...

MySQL ストアド プロシージャで case ステートメントを使用する詳細な例

この記事では、例を使用して、MySQL ストアド プロシージャでの case ステートメントの使用方...

CSS3で作られたレインボーボタンスタイル

結果: 実装コード: html <div class="buttons"&...

Vueカスタムコンポーネントは双方向バインディングを実装します

シナリオ:一般的に使用される親コンポーネントと子コンポーネント間の相互作用方法は次のとおりです。親コ...

MySQL における := と = の違いをグラフィカルに紹介

:= と = の違い=設定および更新の場合にのみ、:= と同じ効果、つまり代入効果があり、それ以外の...

CocosCreator でカメラトラッキングに cc.follow を使用する方法

Cocos Creator バージョン: 2.3.4デモのダウンロード: https://files...

HTML での位置の使用に関する簡単な紹介

昨日 HTML を少し学んだばかりで、JD.com の検索バーを作るのが待ちきれませんでした。 作っ...

Linux システムの busybox に mkfs.vfat コマンドを移植する

オーディオおよびビデオ ファイルを保存するためのディスク寿命を延ばすには、ディスクをフォーマットする...

MySQL 空間データストレージと関数

目次1. データ型1. MySQL空間データとは何か2. GeoJSONとは3. 空間データ型のフォ...

1 分で Nginx のバージョンをスムーズにアップグレードおよびロールバックする方法

今日は、企業の実際の本番環境でよく遭遇する、Nginx を新しいバージョンにアップグレードし、古いバ...

Ubuntu 19.04 インストール チュートリアル (画像とテキストの手順)

1. 準備1.1 VMware 15 をダウンロードしてインストールするダウンロード リンク: h...

MySQL データ型 DECIMAL(N,M) における N と M の意味の詳細な説明

同僚から、MySQL データ型 DECIMAL(N,M) の N と M の意味を尋ねられました。言...