今天在回顧JavaScript進階用法的時候,發現一個有趣的問題,話很少說,先上代碼:javascript
for(var j=0;j<10;j++){
setTimeout(function(){console.log(j)},5000)
}
複製代碼
看到這三行代碼,也許你會不耐煩道:又要講閉包?要吐了好麼?別急,讓咱們先來思考一下,這段代碼在瀏覽器中的執行結果是什麼?java
甲:順序打印0到9?瀏覽器
乙:這題我見過,打印十個10!閉包
哪一個答案正確?咱們繼續上圖:異步
執行結果顯示,瀏覽器打印出了十個10(由於圖片處理的緣由,按下回車到打印以前其實間隔了5秒左右),貌似乙勝出了。但若是你足夠細心,你會發現幾個問題:若是上述三個問題你都能回答上來,恭喜你,你已經開始掌握了JavaScript深層次的知識,若是不能,那就乖乖往下看吧!async
爲何會循環打印十個10函數
許多人習慣用第二個問題中的執行結果來回答這個問題:「for循環執行完跳出以後,纔開始執行setTimeout,因此纔打印了十個10」。這樣的答案,只能說是既應付了本身,又應付了別人。其實,要解答第一個問題,首先要解答的就是第二的問題。性能
你們都知道,JavaScript在ES6出現之前,是沒有塊狀做用域的,這就意味着, 在for循環中用var定義的變量j,實際上是屬於全局的,即在全局範圍內均可以被訪問到,既然如此,那其實整個全局做用域中就只有一個j,每次for循環都i是在更新這個j。優化
那麼如今關鍵的問題在於,爲何整個for循環會先於setTimeout執行,而不是咱們正常理解的,一次迭代執行一次。ui
這就涉及到了JavaScript的核心特性:單線程。
JavaScript設計的初衷,是瀏覽器用來與用戶進行交互和DOM操做的。這就決定了它必須是單線程的,設想JavaScript同事有兩個線程,一個線程在DOM節點添加內容,一個線程刪除該節點,瀏覽器就會出現混亂。因此,爲了不復雜性,從一誕生,JavaScript就是單線程,這已經成了這門語言的核心特徵,未來也不會改變。
單線程就意味着,全部任務須要排隊,前一個任務結束,纔會執行後一個任務。若是前一個任務耗時很長,後一個任務就不得不一直等着。
爲了優化單線程的性能,JavaScript將任務分紅兩種,一種是同步任務(synchronous),另外一種是異步任務(asynchronous)。同步任務指的是,在主線程上排隊執行的任務,只有前一個任務執行完畢,才能執行後一個任務;異步任務指的是,不進入主線程、而進入"任務隊列"(task queue)的任務,只有主線程中的同步任務執行完畢,異步任務纔會進入執行隊列執行。只要主線程空了,就會去讀取"任務隊列",這就是JavaScript的運行機制。這個過程會不斷重複。
而setTimeout,就被JavaScript定義爲異步任務。每次for循環的迭代,都將setTimeout中的回調函數加入任務隊列等待執行。也就是說,只有同步任務中的for循環徹底結束,主線程中才會去任務隊列中找到還沒有執行的十個setTimeout(十次迭代)回調函數並順序執行(先進先出)。而此時,i已經通過循環結束變成了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的深層次問題,雖然總結成一篇文章只有區區數百字,可是我在成文的過程當中查閱了大量的資料,也作了許多實驗。
最後,給出一個很小可是仍然在困擾個人一個問題,但願有興趣的小夥伴能夠跟我一塊兒研究:
setTimeout(function(){while(true){}},6000);
setTimeout(function(){console.log(1)},10000);
setTimeout(function(){console.log(2)},5000);
複製代碼
上述代碼的執行順序是怎樣的?setTimeout的定時,是定時插入執行棧以後當即執行,仍是當即插入執行棧定時執行?
期待你們的留言。