ホームページ ウェブフロントエンド jsチュートリアル JavaScript クロージャー (Closure) の使用例の簡単な分析_JavaScript スキル

JavaScript クロージャー (Closure) の使用例の簡単な分析_JavaScript スキル

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

この記事の例では、JavaScript クロージャー (Closure) の使用法について説明します。参考のために皆さんと共有してください。詳細は次のとおりです:

closureは「閉鎖」と訳されますが、これはあまりにもアカデミックにパッケージ化されているように感じます。書籍やネット上の資料を参考にしながら簡単に説明してみます(不適切な理解に注意してください)。

1. クロージャとは

公式の回答: いわゆる「クロージャ」は、多くの変数とこれらの変数にバインドされた環境を含む式 (通常は関数) を指します。したがって、これらの変数も式の一部です。

上記の定義を読んだ後、もしあなたが専門家でないなら、あなたもきっと私と同じように怒ってこう尋ねると思います。「これはTMDは人間の言語なのか?」
クロージャを理解するには、次のコードが最も説得力があります:

function funcTest()
{
 var tmpNum=100; //私有变量
 //在函数funcTest内定义另外的函数作为funcTest的方法函数
 function innerFuncTest(
 {
    alert(tmpNum); //引用外层函数funcTest的临时变量tmpNum
 }
 return innerFuncTest; //返回内部函数
}
//调用函数
var myFuncTest=funcTest(); 
myFuncTest();//弹出100

ログイン後にコピー

上記のコードでは、コメントがわかりやすく書かれています。これで、「クロージャ」を次のように理解できます。関数本体に別の関数を対象オブジェクトのメソッド関数として定義し(この例では、関数 funcTest 内の funcTest のメソッド関数として別の関数 innerFuncTest を定義します)、メソッド関数このオブジェクトの一時変数を外部関数本体で参照するようになります(クロージャは変数の値を間接的に保持する仕組みです。この例では、内部関数 innerFuncTest は外部関数 funcTest の一時変数 tmpNum を参照します)。ここで、一時変数はすべての宣言されたローカル変数、パラメーター、およびその他の宣言された内部関数に含めることができることに注意してください。これらの内部関数の 1 つが、それらを含む外部関数の外側で呼び出される場合、クロージャーが形成されます (この例では、関数を呼び出すときに、myFuncTest は実際に innerFuncTest 関数を呼び出します。これは、funcTest の内部関数、innerFuncTest が外部で呼び出されるということを意味します) funcTest、クロージャが作成されます)。

2. クロージャを使用する 2 つの例

ここでは 2 つの例を示します。1 つはクロージャが問題を引き起こすため、もう 1 つはクロージャを使用して関数のスコープを通じてパラメータを巧妙にバインドするためです。

これら 2 つの例に関連する HTML マークアップ フラグメントは次のとおりです:

<a href="#" id="closureTest0">利用闭包的例子(1秒后会看到提示)</a><br />
<a href="#" id="closureTest1">由于闭包导致问题的例子1</a><br />
<a href="#" id="closureTest2">由于闭包导致问题的例子2</a><br />
<a href="#" id="closureTest3">由于闭包导致问题的例子3</a><br />

ログイン後にコピー

(1) クロージャによる問題点

上記の HTML タグ フラグメントには 4 つの 要素があります。ここで、ユーザーがクリックしたときにページ内の順序を報告するようにイベント ハンドラーを割り当てる必要があります。 2 番目のリンクにリンクすると、「1 番目のリンクをクリックしました」と報告されます。これを行うには、最後の 3 つのリンクにイベント ハンドラーを追加する次の関数を作成します。

function badClosureExample(){
  for (var i = 1; i <4; i++) {
    var element = document.getElementById('closureTest' + i);
    element .onclick = function(){
      alert('您单击的是第' + i + '个链接');
    }
  }
}

ログイン後にコピー

次に、ページがロードされた後にこの関数を呼び出します (そうでないとエラーが報告される可能性があります):

window.onload = function(){
  badClosureExample();
}

ログイン後にコピー

最後の 3 つのリンクをクリックすると、実行結果を確認します。警告ボックスにはどのような情報が表示されますか? ——すべて「4番目のリンクをクリックした」だけです。驚きますか?なぜ?

分析: badClosureExample() 関数の element.onclick に割り当てられたイベント ハンドラー、つまり、onclick 匿名関数は、 badClosureExample() 関数の完了後 (ユーザーがリンクをクリックしたとき) にのみ呼び出されるためです。呼び出し時には、変数 i を評価する必要があります。パーサーは最初にイベント ハンドラー内を検索しますが、i は定義されていません。次に、 badClosureExample() 関数を検索します。この時点では定義されていますが、i の値は 4 です (i が 4 より大きい場合にのみ for ループの実行が停止します)。したがって、その値が取得されます。これは、クロージャ (匿名関数) がその外部関数 (badClosureExample) のスコープ内で変数を使用する場合に実行される動作とまったく同じです。さらに、これは、匿名関数自体がパラメーターを渡すことができない (したがって、独自のスコープを維持できない) という事実によっても引き起こされます。

それでは、この例の問題をどのように解決すればよいでしょうか?実際には、多くのメソッドがあります (自分で書いて見てみるのも良いでしょう)。コードは比較的単純で直接的だと思います。

function popNum(oNum){
  return function(){
          alert('您单击的是第'+oNum+'个链接');
  }
}
function badClosureExample(){
  for (var i = 1; i <4; i++) {
    var element = document.getElementById('closureTest' + i);
    element .onclick =new popNum(i);
    }
}

ログイン後にコピー

(2)、パラメータをバインドするためのクロージャの賢明な使用法

上記の HTML フラグメントでは、ユーザーが最初のリンクをクリックしたときに警告ボックスが表示されるのを遅らせたいと考えています。これを実現するにはどうすればよいでしょうか。答えは、setTimeout() 関数を使用することです。この関数は、次のような、指定されたミリ秒数の後に関数を呼び出します。

コードをコピー コードは次のとおりです:
setTimeout(someFunc,1000);
しかし、問題はパラメータを someFunc 関数に渡すことができないことです。この問題はクロージャを使用すると簡単に解決できます:

function goodClosureExample(oMsg){
  return function(){
    alert(oMsg);
  };
}

ログイン後にコピー
関数 GoodClosureExample は、匿名関数 (クロージャ) を返すために使用されます。そして、次のようなパラメータを渡すことで、返された匿名関数にパラメータをバインドさせることができます。


コードをコピー コードは次のとおりです。var Good = GoodClosureExample('このパラメータはクロージャを通じてバインドされています') ;

而此时,就可以将绑定了参数的good函数传递给setTimeout()实现延时警告了:
复制代码 代码如下:
setTimeout(good,1000) //此时good中已经绑定了参数

最后,测试通过的完整代码:

window.onload = function(){
  var element = document.getElementById('closureTest0');
  if (element) {
    var good = goodClosureExample('这个参数是由闭包绑定的');
    element.onclick = function(){
      setTimeout(good, 1000); //延迟1秒弹出提示
    }
  }
}

ログイン後にコピー

3、javascript的垃圾回收原理

(1)、在javascript中,如果一个对象不再被引用,那么这个对象就会被GC回收;

(2)、如果两个对象互相引用,而不再被第3者所引用,那么这两个互相引用的对象也会被回收。

在js中使用闭包,往往会给javascript的垃圾回收器制造难题。尤其是遇到对象间复杂的循环引用时,垃圾回收的判断逻辑非常复杂,搞不好就有内存泄漏的危险,所以,慎用闭包。ms貌似已经不建议使用闭包了。

希望本文所述对大家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衣類リムーバー

AI Hentai Generator

AI Hentai Generator

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. クロージャによるメモリリーク クロージャの特徴は、内部関数が外部関数の変数にアクセスできることです。つまり、クロージャ内で参照される変数はガベージコレクションされません。不適切に使用すると、

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

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

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

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

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

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

golangの匿名関数とクロージャの長所と短所のまとめ golangの匿名関数とクロージャの長所と短所のまとめ May 05, 2024 am 09:54 AM

匿名関数は簡潔で匿名ですが、可読性が低く、デバッグが困難です。クロージャはデータをカプセル化して状態を管理できますが、メモリの消費と循環参照が発生する可能性があります。実際のケース: 匿名関数は単純な数値処理に使用でき、クロージャは状態管理を実装できます。

See all articles