• <noscript id="ggggg"><dd id="ggggg"></dd></noscript>
    <small id="ggggg"></small> <sup id="ggggg"></sup>
    <noscript id="ggggg"><dd id="ggggg"></dd></noscript>
    <tfoot id="ggggg"></tfoot>
  • <nav id="ggggg"><cite id="ggggg"></cite></nav>
    <nav id="ggggg"></nav>
    成人黃色A片免费看三更小说,精品人妻av区波多野结衣,亚洲第一极品精品无码,欧美综合区自拍亚洲综合,久久99青青精品免费观看,中文字幕在线中字日韩 ,亚洲国产精品18久久久久久,黄色在线免费观看

    你不知道的JavaScript:有趣的setTimeout

    2018-5-7    周周

         話不多說,先上代碼

         for(var j=0;j<10;J++){

             setTimeout(function(){console.log(j);},5000)

         }

         看到這三行代碼,你也許會不耐煩道:又要講閉包?要吐槽了好么?別急,讓我們先來思考一下,這段代碼在瀏覽器中的執行結果是什么?

         <!-- more -->

         甲:順序打印0到9?

         乙:這題我見過,打印十個10!

         哪個答案正確?

         執行結果顯示,瀏覽器打印出十個10,貌似乙對了,但是如果你足夠細心,你會發現幾個問題:為什么會循環打印十個10,而不是0到9?

         從結果來看,for循環執行完跳出之后,才開始執行setTimeout(所以j才等于10),為什么不是每次迭代都執行一次setTimeout呢?

    1、為什么會循環打印十個10?

          許多人習慣用第二個問題中的執行結果來回答這個問題:“for循環執行完畢跳出之后才開始執行setTimeout,所以才打印了十個10”。這樣的答案,只能說是既應付了自己,又應付了別人。其實,要解答第一個問題,首先要解答第二個問題。

    2、為什么不是每一次迭代都執行一次setTimeout?

         大家都知道,JavaScript在ES6出現以前,是沒有塊狀作用域的,這就意味著,在for循環中用var定義的變量j,其實是屬于全局的,那其實整個全局作用域中只有一個j,每次for循環都是更新這個j。

           那么現在的關鍵問題在于,為什么整個for循環會先于setTimeout執行,而不是我們正常理解的,一次迭代執行一次。這就涉及到了JavaScript的核心特性:單線程。

           JavaScript設計的初衷,是瀏覽器用來與用戶進行交互和DOM操作的,這就決定了它必須是單線程的。設想JavaScript同時有兩個線程,一個線程在DOM節點內添加內容,一個線程刪除該節點,瀏覽器就會出現混亂。所以,為了避免復雜性,從一誕生,JavaScript就是單線程,這已經成為了這門語言的核心特征,將來也不會改變。

          單線程就意味著,所有任務需要排隊,前一個任務結束,才會執行下一個任務,如果前一個任務耗時很長,后一個任務就不得不一直等著。

          為了優化單線程的性能,JavaScript將任務分成兩種,一種是同步任務(synchronous),另一種是異步任務(asynchronous)。同步任務指的是,在主線程上排隊執行的任務,只有前一個任務執行完畢,才能執行后一個任務;異步任務指的是,不進入主線程,而進入“任務隊列(task queue)”的任務,只用主線程中的同步任務執行完畢,異步任務才會進入執行隊列執行。只要主線程空了,就會去讀取“任務隊列”,這就是JavaScript的運作機制。這個過程會不斷重復。

           而setTimeout就被JavaScript定義為異步任務。每次for循環的迭代,都將setTimeout中的回調函數加入任務隊列等待執行。也就是說,只有同步任務中的for循環完全結束,主線程中才會去任務列表中找到尚未執行的十個setTimeout(十次迭代)回調函數并順序執行(先進先出)。而此時,j已經經過循環結束變成了10,所以此時主線程執行的,是十個一模一樣的打印i的回調函數,即打印十個10,。至此完美回答了第一和第二個問題,文章開頭的代碼與下面的代碼其實是等價的:

           for(var i=0;i<10;i++){

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

                setTimeout(console.log(i),5000);

    }

           小小的一個setTimeout,牽扯出了很多JavaScript的深層次問題,可見JavaScript還有許多地方是值得深入探究的。



    日歷

    鏈接

    個人資料

    藍藍設計的小編 http://www.lzhte.cn

    存檔

    主站蜘蛛池模板: 日韩亚洲AV最新在线观看| 在线a人片免费观看国产| 熟妇人妻一区二区三区四区| 成人国产在线看不卡| 韩日成人| 牛牛澡牛牛爽一区二区| 老妇毛片久久久久久久久| 五月综合婷婷开心综合婷婷| 内地偷拍一区二区三区| 亚洲一区二区精品推荐| 老子影院午夜精品无码| 精品九九人人做人人爱| 久久精品岛国av一区二区无码| 久久999精品国产只有精品| 国产免费久久久久久无码| 伊大人香蕉久久网欧美| 沾化县| 999人在线精品播放视频| 粗大猛烈进出高潮视频无码| 亚洲日本国产综合高清醉红楼| 欧美另类视频在线观看| 久久综合九色综合精品| 国产精品天干在线观看| 欧美在线观看免费做受视频| 欧美大胆A级视频免费| 日本人又色又爽的视频| 亚洲免费观看一区二区三区| 亚洲国产成人无码网站大全| 国产成人综合亚洲欧洲色就色| 首页亚洲国产丝袜长腿综合 | 国产真实伦视频在线视频| 久久福利影院| 青川县| 国产日韩精品视频无码| 亚洲永久精品一区二区三区| 精品国精品国产自在久国产应用| 一本大道香蕉久中文在线播放| 午夜无遮挡男女啪啪免费软件| 九江县| 亚洲日本无码一区二区在线观看| 三级黄片一区二区三区|