關於for循環中使用setTimeout的四種解決方案


  我們先來簡單了解一下setTimeout延時器的運行機制。setTimeout會先將回調函數放到等待隊列中,等待區域內其他主程序執行完畢后,按時間順序先進先出執行回調函數。本質上是作用域的問題。

  因此若是這樣將不會得到想要的結果輸出1.2.3.4.5,而會連續輸出5個6。

for (var i=1; i<=5; i++) {
    setTimeout( function timer() {
        console.log( i );
     }, i*1000 );
}

  這是因為setTimeout是異步執行,每一次for循環的時候,setTimeout都執行一次,但是里面的函數沒有被執行,而是被放到了任務隊列里,等待執行。只有主線上的任務執行完,才會執行任務隊列里的任務。也就是說它會等到for循環全部運行完畢后,才會執行fun函數,但是當for循環結束后此時i的值已經變成了6,因此雖然定時器跑了5秒,控制台上的內容依然是6。

  (注意:for循環從開始到結束的過程,需要維持幾微秒或幾毫秒,當定時器跑完一秒之后for循環早已經做完了。)

  我們來看另一種情況:

for (var i=1; i<=5; i++) {
    (function() {
        setTimeout( function timer() {
            console.log( i );
        }, i*1000 );
    })();
}

  由setTimeout的運行機制可以知道,首先會運行外部的所有主程序,雖然for循環內形成了閉包,但是fun並沒有發現一個實參所以跟第一個例子並無實際差別,仍然是連續輸出5個6。

解決方案1:閉包

  使用閉包是很經典的一種做法:

for (var i=1; i<=5; i++) {
    (function(j) {
        setTimeout( function timer() {
            console.log( j );
        }, j*1000 );
    })(i);
}

  我們可以發現跟預期結果一致,依次輸出1到5,因是因為實際參數跟定時器內部的i有強依賴。

  通過閉包,將i的變量駐留在內存中,當輸出j時,引用的是外部函數的變量值i,i的值是根據循環來的,執行setTimeout時已經確定了里面的的輸出了。

解決方案2:拆分結構

  我們還可以將setTimeout的定義和調用分別放到不同部分:

function timer(i) {
    setTimeout( console.log( i ), i*1000 );
}
for (var i=1; i<=5;i++) {
    timer(i);
}

  控制台上輸出依然是依次輸出1到5。

解決方案3:let

  這里再來說一說使用es6的let來解決此問題:

for (let i=1; i<=5; i++) {
    setTimeout( function timer() {
        console.log( i );
     }, i*1000 );
}

  這個例子與第一個相比,只是把var更改成了let,可是控制台的結果卻是依次輸出1到5。

  因為for循環頭部的let不僅將i綁定到for循環中,事實上它將其重新綁定到循環體的每一次迭代中,確保上一次迭代結束的值重新被賦值。setTimeout里面的function()屬於一個新的域,通過var定義的變量是無法傳入到這個函數執行域中的,通過使用let來聲明塊變量能作用於這個塊,所以function就能使用i這個變量了;這個匿名函數的參數作用域和for參數的作用域不一樣,是利用了這一點來完成的。這個匿名函數的作用域有點類似類的屬性,是可以被內層方法使用的。

解決方案4:setTimeout第三個參數

for (let i=1; i<=5; i++) {
    setTimeout( function timer() {
        console.log( i );
     }, i*1000, i );
}

  由於每次傳入的參數是從for循環里面取到的值,所以會依次輸出1到5。關於setTimeout第三個參數,下一篇會詳細講到,這里大家了解下就好。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM