ホームページ > ウェブフロントエンド > jsチュートリアル > 閉鎖とは何ですか? JavaScript のクロージャについて話して、どのような機能があるかを見てみましょう。

閉鎖とは何ですか? JavaScript のクロージャについて話して、どのような機能があるかを見てみましょう。

青灯夜游
リリース: 2022-07-08 11:14:54
転載
1734 人が閲覧しました

クロージャとは何ですか?クロージャーが何をするかわかりますか?次の記事では、JavaScript のクロージャについて説明します。一定の参考値があるので、困っている友達が参考になれば幸いです。

閉鎖とは何ですか? JavaScript のクロージャについて話して、どのような機能があるかを見てみましょう。

フロントエンドの学習の過程では、必然的に多くの問題に遭遇します。そこで、今日は初心者の観点から 2 つの問題について説明します。

クロージャとは何ですか?

クロージャの機能は何ですか?

実際、JavaScript を学習するとき、クロージャはどこにでもあります。必要なのは、それを認識して受け入れることができることだけです。クロージャは、使用する新しい構文やパターンを学習する必要があるツールではなく、字句スコープに基づいてコードを作成することの自然な結果です。コードを記述するときに意図的にクロージャを作成する必要はほとんどありません。

すでに多くの友人が心の中でつぶやいていると思いますが、この語彙範囲は何ですか? パニックにならないで、ゆっくり聞いてください。つまり、語彙範囲は語彙範囲で定義されます。 。つまり、字句スコープは、コードを記述するときに変数とブロックレベルのスコープを配置する場所によって決定されるため、字句アナライザーがコードを処理するとき(ほとんどの場合)、スコープは変更されません。 ——「あなたの知らない JavaScript のボリューム」

最初に例を見てみましょう

function test(){
   var arr = []
   for(var i=0;i<10;i++){
       arr[i]=function(){
           console.log(i);
       }
   }
   return arr
}

var myArr = test()
// myArr[0]()
// myArr[1]()
// ...

for(var j = 0; j < 10; j++){ 
   myArr[j]()
}
//为了避免繁琐此处使用了第二个循环来调用test函数里第一个循环里函数的打印出十个结果
ログイン後にコピー

最初にこのコードを分析してみましょう。 このコードが実行されると、常識的な分析によれば、0 から 9 までの 10 個の数字が順番に出力されるはずですが、for ループの実行には時間がかかりません (マイクロ秒単位で無視されます)。 ] 中には function(){console.log(i);} が 10 個あります。このとき、配列内の関数は実行されません。var myArr = test() がテスト関数を呼び出すと、for ループの実行時間が経過するため、は無視されます。このとき、i はすでに 10 なので、10 が 10 個表示されます。

この時点で誰かが尋ねると思います、これはこれから説明するクロージャと何の関係があるのですか? では、このコードを少し変更してアキュムレータに変更したらどうなるでしょうか?彼に気づいていますか?

現時点では、そんな簡単なことではないと言う大物もいると思います。

最初の for ループがブロック レベルのスコープになるように、var 定義を let 定義に変更し、その後、アキュムレータにできるようにします。もちろん問題はありません。

しかし、今日私たちが話しているのは、ES5 にアキュムレーターを実装する方法です。次に、次のコードを見てみましょう:

function test(){
    var arr = []
    for(var i=0;i<10;i++){
        (function(j){
            arr[j]=function(){
                console.log(j);
            }
        })(i)
    }
    return arr
}

var myArr = test()

for(var j = 0; j < 10; j++){ 
    myArr[j]()
}
ログイン後にコピー

注意深い人なら、これがループ内の関数本体を自己実行関数に変更していることに間違いなく気づくでしょうが、このときの出力結果は次のとおりです。 クロージャを含む0から9までの10個の数値が順番に出力されます。このコードの実行を開始すると、2番目のforループが10回呼び出され、それぞれの自己実行関数が実行されると、自己実行関数が作成されます。実行関数の AO オブジェクト。自己実行関数の AO オブジェクトには j という名前の属性があります。通常、自己実行関数の実行後、その AO オブジェクトは破棄されるはずですが、myarr[j] ( ) を実行すると、スコープチェーンの先頭にある arr[j] の AO オブジェクト内で属性名 j が検索されますが見つかりません。次に、スコープチェーンを下に向かって検索し、AO オブジェクト内で属性名 j を見つけます。実行関数が終了すると、その AO オブジェクトはガベージ コレクション メカニズムによってリサイクルされません。そうでない場合は、myarr[j] () の実行時にエラーが報告されます。この時点でクロージャが形成されます。

関数が外部に保存されると、クロージャが生成されます

別の例を挙げてみましょう

function a(){
    function b(){
        var bbb = 234
        console.log(aaa);
    }
    var aaa = 123
    return b // b出生在a里面,但是被保存出去了
}

var glob = 100
var demo = a()
demo()
ログイン後にコピー

このコードでは、まず、プリコンパイルを使用して分析します まず、グローバル GO オブジェクトを定義します グローバル宣言を見つけて、グローバル変数宣言を見つけます 変数宣言を GO の属性名として使用し、値は未定義です グローバルに関数宣言を見つけて関数名を使用しますas GO オブジェクトの属性名。値は関数本体に割り当てられます。この時点では、 GO{ glob: unknown--->100; demo: unknown; a: fa(){} }; となるはずです。 次に、関数 a の AO{ aaa: unknown--->123;b: fb(){} } を作成し、最後に関数 a で関数 b をプリコンパイルして AO{ b: unknown--->234} を作成します。 ; このときのスコープチェーンの順序は、 1. 関数bのAOオブジェクト; 2. 関数aのAOオブジェクト; 3. グローバルGOオブジェクトです。関数 b で aaa を出力するときは、スコープ チェーンの先頭から開始します。関数 b の AO オブジェクトに aaa がない場合は、スコープ チェーンに沿って下方向に検索して、第 2 レベルの関数 a の AO を見つけます。目的は、aaa の値が 123 であることを見つけ、結果を出力することです。

如果我们没有从预编译的角度去分析就会认为此时的aaa应该会报错的,当var demo = a()执行时,当a函数执行结束,那么a对应的AO对象应该被销毁了,照常理分析当我们执行demo时作用域链此时应该会创建b的AO对象和GO对象,此时只有b的AO对象,没有a的AO对象,应该不能打印出aaa的值,但是此时aaa的值为123,则说明a的AO对象没有被销毁,那么为什么呢?原因就在于这里创建了闭包,当var demo = a()执行结束之后,垃圾回收机制会来问,a函数老兄,我看你都执行完毕了,你的运行内存是不是可以给我释放了,但是此时a函数只能无奈摇摇头说道,老哥,我也不确定我有没有执行完毕,我执行完创建了一个b,但是b又不归我管,所以我也不确定b有没有被调用,所以我也不确定我有没有执行完毕,垃圾回收机制想了想,既然你都不知道那我就不回收了,要是回收了还没执行完的就该报错了,所以此时a的AO对象就没有被回收。

补充全面一点就是:当一个函数内部的函数被保存到函数外部时,就会产生闭包。

相信通过这两个例子,你已经对闭包有了一个大概的了解,那接下来我们说一下闭包有哪些作用。

闭包的作用

    1. 实现公有变量 例如:累加器(3.js)
    1. 做缓存
    1. 可以实现封装,属性私有化
    1. 模块化开发,防止污染全局变量

我们对闭包的作用也来一个例子(3.js)

 var count = 0
 function add() {
   return count++
 }
 console.log(add());
 console.log(add());
 console.log(add());
ログイン後にコピー

这是一段比较普通的累加的代码,但是如果我们在实习甚至是工作的时,公司要求你把累加器封装成一个模块化的代码,那么
此时,为了模块化我们尽可能避免定义全局变量,但是不定义全局变量我们如何实现呢,此时我们就可以用到闭包了;

function add() {
    var count = 0
    function a() {
      ++count
      console.log(count);
    }
    return a
  }
  var res = add()
  res()
  res()
  
  //add函数结束之后,add的AO对象没有被销毁,因为add函数执行完了之后,返回的a不知道是否被调用就形成了闭包,这样
  就能使得不使用全局变量也能封装成一个模块化的累加器。
ログイン後にコピー

结语

那么关于闭包以及闭包的作用相关的一些个人见解就是这些,目前对于闭包也只是一些浅显的了解,后期学习之后完善过后会出后续关于闭包的相关文章,感谢您的观看,欢迎批评斧正,一起共同进步。

【相关推荐:javascript视频教程web前端

以上が閉鎖とは何ですか? JavaScript のクロージャについて話して、どのような機能があるかを見てみましょう。の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
ソース:juejin.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート