ホームページ ウェブフロントエンド jsチュートリアル JavaScript 関数型プログラミングのクロージャを理解する_JavaScript スキル

JavaScript 関数型プログラミングのクロージャを理解する_JavaScript スキル

May 16, 2016 pm 03:11 PM
javascript 閉鎖

クロージャは、1960 年代に登場した関数型プログラミングの概念で、クロージャを実装した最初の言語は LISP の方言である Scheme でした。それ以来、クロージャ機能は他の言語で広く採用されてきました。
クロージャの厳密な定義は、「関数 (環境) とそれに囲まれた自由変数で構成されるセット」です。この定義は誰にとっても少しわかりにくいため、最初に例を通して説明し、それほど厳密ではないクロージャを説明します。次に、クロージャの古典的な使用例をいくつか示します。

クロージャとは

平たく言えば、JavaScript のすべての関数はクロージャですが、一般に、ネストされた関数は
をより適切に反映できます。 クロージャーの特性を示すために、次の例を参照してください:

var generateClosure = function() {
var count = 0;
var get = function() {
count ++;
return count;
};
return get;
};
var counter = generateClosure();
console.log(counter()); // 输出 1
console.log(counter()); // 输出 2
console.log(counter()); // 输出 3
ログイン後にコピー

このコードでは、generateClosure() 関数にローカル変数 count があり、初期値は 0 です。 get と呼ばれる関数もあります。この関数は、親スコープであるgenerateClosure() 関数内の count 変数を 1 ずつ増加させ、count の値を返します。 generateClosure() の戻り値は get 関数です。外部的には、カウンター変数を介してgenerateClosure()関数を呼び出し、その戻り値であるget関数を取得しました。その後、counter()を数回繰り返し呼び出したところ、戻り値が毎回1ずつ増加することがわかりました。
上記の例の特徴を見てみましょう。命令型プログラミングの考え方の通常の理解によれば、そのライフサイクルは、generateClosure がコール スタックから呼び出されるときの期間です。カウント変数 適用されていたスペースが解放されます。問題は、generateClosure() の呼び出しが終了した後、counter() が「すでに解放された」カウント変数を参照し、エラーが発生しないだけでなく、counter() が呼び出されるたびに count が変更されて返されることです。どうしたの?
これはまさにいわゆるクロージャの特徴です。関数がその中で定義された関数を返す場合、クロージャには返された関数だけでなく、関数が定義されている環境も含まれます。上記の例では、関数generateClosure()の内部関数getが外部変数counterから参照されている場合、counterとgenerateClosure()のローカル変数がクロージャとなります。十分に明確でない場合は、次の例が役立つかもしれません
あなたは理解しています:

var generateClosure = function() {
var count = 0;
var get = function() {
count ++;
return count;
};
return get;
};
var counter1 = generateClosure();
var counter2 = generateClosure();
console.log(counter1()); // 输出 1
console.log(counter2()); // 输出 1
console.log(counter1()); // 输出 2
console.log(counter1()); // 输出 3
console.log(counter2()); // 输出 2
ログイン後にコピー

上記の例は、クロージャがどのように生成されるかを説明しています。counter1 と counter2 はそれぞれ、generateClosure() 関数を呼び出し、クロージャの 2 つのインスタンスを生成します。内部的に参照する count 変数は、それぞれの動作環境に属します。 generateClosure() が get 関数を返すと、get が参照する可能性のあるgenerateClosure() 関数の内部変数 (つまり、count 変数) もプライベートに返され、メモリ内にコピーが生成されることがわかります。 thengenerateClosure() 返された関数の 2 つのインスタンス、counter1 と counter2 は互いに独立しています。

閉鎖の目的

1. ネストされたコールバック関数
クロージャには主に 2 つの用途があります。1 つはネストされたコールバック関数を実装すること、もう 1 つはオブジェクトの詳細を隠すことです。まず、ネストされたコールバック関数を理解するために次のコード例を見てみましょう。次のコードは、Node.js で MongoDB を使用して、ユーザーを追加する単純な関数を実装します。

exports.add_user = function(user_info, callback) {
var uid = parseInt(user_info['uid']);
mongodb.open(function(err, db) {
if (err) {callback(err); return;}
db.collection('users', function(err, collection) {
if (err) {callback(err); return;}
collection.ensureIndex("uid", function(err) {
if (err) {callback(err); return;}
collection.ensureIndex("username", function(err) {
if (err) {callback(err); return;}
collection.findOne({uid: uid}, function(err) {
if (err) {callback(err); return;}
if (doc) {
callback('occupied');
} else {
var user = {
uid: uid,
user: user_info,
};
collection.insert(user, function(err) {
callback(err);
});
}
});
});
});
});
});
};
ログイン後にコピー
Node.js や MongoDB に詳しくなくても問題ありません。詳細を理解する必要はなく、一般的なロジックを理解するだけで十分です。このコードはクロージャのネストの層を使用しており、ネストの各層はコールバック関数です。コールバック関数はすぐには実行されませんが、対応するリクエストが処理された後に、リクエストされた関数によってコールバックされます。ネストの各層にコールバックへの参照があり、最も内側の層でも外側の層で定義された uid 変数が使用されていることがわかります。クロージャ メカニズムの存在により、外部関数が実行された場合でも、そのスコープに適用された変数は解放されません。これは、内部関数が引き続きこれらの変数を参照する可能性があるためです。これにより、ネストされた非同期コールバックが完全に実現されます。

2. プライベートメンバーの実装 JavaScript オブジェクトにはプライベート プロパティがないことがわかっています。これは、オブジェクトのすべてのプロパティが外部に公開されていることを意味します。これにより、たとえば、オブジェクトのユーザーが属性を直接変更すると、オブジェクトの内部データの一貫性が破壊されるなど、セキュリティ リスクが発生する可能性があります。 JavaScript では、このプロパティがプライベートであり、外部オブジェクトが直接読み書きできないことを示すために、すべてのプライベート プロパティ (_myPrivateProp など) の前にアンダースコアを置く規則が使用されています。しかし、これは単なる非公式の合意であり、オブジェクトのユーザーがこれを行わないと仮定すると、より厳密なメカニズムはあるのでしょうか。答えは「はい」です。クロージャを通じて実現できます。前の例をもう一度見てみましょう:

var generateClosure = function() {
var count = 0;
var get = function() {
count ++;
return count;
};
return get;
};
var counter = generateClosure();
console.log(counter()); // 输出 1
console.log(counter()); // 输出 2
console.log(counter()); // 输出 3
ログイン後にコピー

counter() を呼び出すことのみがクロージャ内の count 変数にアクセスし、ルールに従って 1 ずつインクリメントできることがわかります。他の方法で count 変数を見つけることは不可能です。この単純な例から着想を得て、オブジェクトをクロージャでカプセル化し、詳細を隠すために「アクセサ」オブジェクトのみを返すことができます。

上記がこの記事の全内容です。これが皆さんの JavaScript クロージャの学習と理解に役立つことを願っています。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

C++ ラムダ式におけるクロージャの意味は何ですか? C++ ラムダ式におけるクロージャの意味は何ですか? Apr 17, 2024 pm 06:15 PM

C++ では、クロージャは外部変数にアクセスできるラムダ式です。クロージャを作成するには、ラムダ式の外部変数をキャプチャします。クロージャには、再利用性、情報の隠蔽、評価の遅延などの利点があります。これらは、イベント ハンドラーなど、外部変数が破棄されてもクロージャが外部変数にアクセスできる現実の状況で役立ちます。

C++ラムダ式でクロージャを実装するにはどうすればよいですか? C++ラムダ式でクロージャを実装するにはどうすればよいですか? Jun 01, 2024 pm 05:50 PM

C++ ラムダ式は、関数スコープ変数を保存し、関数からアクセスできるようにするクロージャーをサポートしています。構文は [キャプチャリスト](パラメータ)->戻り値の型{関数本体} です。 Capture-list は、キャプチャする変数を定義します。[=] を使用してすべてのローカル変数を値によってキャプチャするか、[&] を使用してすべてのローカル変数を参照によってキャプチャするか、[variable1, variable2,...] を使用して特定の変数をキャプチャできます。ラムダ式はキャプチャされた変数にのみアクセスできますが、元の値を変更することはできません。

C++ 関数におけるクロージャの長所と短所は何ですか? C++ 関数におけるクロージャの長所と短所は何ですか? Apr 25, 2024 pm 01:33 PM

クロージャは、外部関数のスコープ内の変数にアクセスできる入れ子関数です。その利点には、データのカプセル化、状態の保持、および柔軟性が含まれます。デメリットとしては、メモリ消費量、パフォーマンスへの影響、デバッグの複雑さなどが挙げられます。さらに、クロージャは匿名関数を作成し、それをコールバックまたは引数として他の関数に渡すことができます。

クロージャによって引き起こされるメモリリークの問題を解決する クロージャによって引き起こされるメモリリークの問題を解決する Feb 18, 2024 pm 03:20 PM

タイトル: クロージャによって引き起こされるメモリ リークと解決策 はじめに: クロージャは JavaScript における非常に一般的な概念であり、内部関数が外部関数の変数にアクセスできるようにします。ただし、クロージャを誤って使用すると、メモリ リークが発生する可能性があります。この記事では、クロージャによって引き起こされるメモリ リークの問題を調査し、解決策と具体的なコード例を提供します。 1. クロージャによるメモリリーク クロージャの特徴は、内部関数が外部関数の変数にアクセスできることです。つまり、クロージャ内で参照される変数はガベージコレクションされません。不適切に使用すると、

PHP 関数の連鎖呼び出しとクロージャ PHP 関数の連鎖呼び出しとクロージャ Apr 13, 2024 am 11:18 AM

はい、コードの単純さと読みやすさは、連鎖呼び出しとクロージャーによって最適化できます。連鎖呼び出しは、関数呼び出しを流暢なインターフェイスにリンクします。クロージャは再利用可能なコード ブロックを作成し、関数の外の変数にアクセスします。

関数ポインタとクロージャが Golang のパフォーマンスに与える影響 関数ポインタとクロージャが Golang のパフォーマンスに与える影響 Apr 15, 2024 am 10:36 AM

関数ポインタとクロージャが Go のパフォーマンスに与える影響は次のとおりです。 関数ポインタ: 直接呼び出しよりわずかに遅くなりますが、可読性と再利用性が向上します。クロージャ: 一般に遅いですが、データと動作をカプセル化します。実際のケース: 関数ポインターは並べ替えアルゴリズムを最適化でき、クロージャーはイベント ハンドラーを作成できますが、パフォーマンスの低下をもたらします。

テストにおける golang 関数クロージャの役割 テストにおける golang 関数クロージャの役割 Apr 24, 2024 am 08:54 AM

Go 言語の関数クロージャは単体テストで重要な役割を果たします。 値のキャプチャ: クロージャは外側のスコープ内の変数にアクセスできるため、テスト パラメータをキャプチャしてネストされた関数で再利用できます。テスト コードの簡素化: クロージャは値をキャプチャすることで、ループごとにパラメーターを繰り返し設定する必要性を排除し、テスト コードを簡素化します。可読性の向上: クロージャを使用してテスト ロジックを整理し、テスト コードをより明確で読みやすくします。

Java ではクロージャはどのように実装されますか? Java ではクロージャはどのように実装されますか? May 03, 2024 pm 12:48 PM

Java のクロージャを使用すると、外部関数が終了した場合でも、内部関数が外部スコープの変数にアクセスできるようになります。匿名の内部クラスを通じて実装されると、内部クラスは外部クラスへの参照を保持し、外部変数をアクティブに保ちます。クロージャによりコードの柔軟性が向上しますが、匿名の内部クラスによる外部変数への参照により、それらの変数が存続するため、メモリ リークのリスクに注意する必要があります。

See all articles