隨著對JS的深入,異步編程是每個JS程序員都跳不過的話題炮车。不僅在前端開發(fā)有大量的異步事件處理衷笋,NODE更是有出了名的callback hell。為了更好地掌握J(rèn)S異步編程豫领,讓我們從基礎(chǔ)開始看起吧抡柿。
先看一個簡單的例子:
for (var i = 1; i <= 3; i++) {
setTimeout(function() { console.log(i); }, 0);
}
若是從C、JAVA轉(zhuǎn)來的程序員等恐,第一直覺是setTimeout延時為0洲劣,因此程序應(yīng)該打印1,2课蔬,3囱稽。但是程序執(zhí)行的結(jié)果卻是4,4二跋,4战惊。
再看下面一個列子:
var start = new Date;
setTimeout(function(){
var end = new Date;
console.log('Time elapsed:', end - start, 'ms');
}, 500);
while (new Date - start < 1000) {};
SetTimeout設(shè)置了500ms的延遲,而隨后的while則空循環(huán)等待1000ms扎即。直覺的反應(yīng)是console會打印500吞获,但執(zhí)行的結(jié)果卻是出乎意料的約等于1000。
這里充分說明了一個問題:setTimeout沒有調(diào)用另一個線程谚鄙,JS程序的執(zhí)行是基于單線程模型的各拷。于是當(dāng)調(diào)用setTimeout的時候,這個延時事件便會排入事件隊列闷营。而線程繼續(xù)執(zhí)行后續(xù)的代碼烤黍,直到代碼全部執(zhí)行完畢之后才來處理事件隊列中的事件。
處理用戶輸入時也是類似的情況,當(dāng)用戶點擊一個附加了事件處理函數(shù)的對象時速蕊,這個點擊事件就排入事件隊列嫂丙。但該事件并不會被立即處理,而是要等到當(dāng)前代碼全部運行結(jié)束后才會被處理互例。
小結(jié):JS是單線程執(zhí)行的奢入,JS代碼永遠(yuǎn)不會被中斷筝闹,所有的事件都會被排入事件隊列媳叨,直到代碼執(zhí)行完畢后才會處理這些事件。
P.S: webkit的console.log也是異步的关顷,在瀏覽器中執(zhí)行:
var obj = {};
console.log(obj);
obj.foo = 'bar';
console并沒有立即打印空對象糊秆,而是等到代碼返回到事件隊列的時候才打印對象,此時打印是對象{foo:bar}议双。
與此相反痘番,NODE中console是同步的,將會立即打印{}平痰。