JavaScrip單線程引擎工作原理分析
更新時(shí)間:2010年09月04日 20:11:53 作者:
從基礎(chǔ)的層面來(lái)講,理解JavaScript的定時(shí)器是如何工作的是非常重要的,文章通過(guò)講解JavaScript的單線程引擎,從而讓讀者更加深入的了解JavaScript定時(shí)器。
從基礎(chǔ)的層面來(lái)講,理解JavaScript的定時(shí)器是如何工作的是非常重要的。定時(shí)器的執(zhí)行常常和我們的直觀想象不同,那是因?yàn)镴avaScript引擎是單線程的。我們先來(lái)認(rèn)識(shí)一下下面三個(gè)函數(shù)是如何控制計(jì)時(shí)器的。
腳本之家推薦閱讀:雕蟲無(wú)小技 JavaScript初學(xué)者的10個(gè)迷你技巧
var id = setTimeout(fn, delay);
初始化一個(gè)計(jì)時(shí)器,然后在指定的時(shí)間間隔后執(zhí)行。該函數(shù)返回一個(gè)唯一的標(biāo)志ID(Number類型),我們可以使用它來(lái)取消計(jì)時(shí)器。
var id = setInterval(fn, delay);
和setTimeout有些類似,但它是連續(xù)調(diào)用一個(gè)函數(shù)(時(shí)間間隔是delay參數(shù))直到它被取消。
clearInterval(id);, clearTimeout(id);
使用計(jì)時(shí)器ID(setTimeout 和 setInterval的返回值)來(lái)取消計(jì)時(shí)器回調(diào)的發(fā)生。
為了理解計(jì)時(shí)器的內(nèi)在執(zhí)行原理,有一個(gè)重要的概念需要加以探討:計(jì)時(shí)器的延遲(delay)是無(wú)法得到保障的。由于所有JavaScript代碼是在一個(gè)線程里執(zhí)行的,所有異步事件(例如,鼠標(biāo)點(diǎn)擊和計(jì)時(shí)器)只有擁有執(zhí)行機(jī)會(huì)時(shí)才會(huì)執(zhí)行。用一個(gè)很好的圖表加以說(shuō)明:

在這個(gè)圖表中有許多信息需要理解,如果完全理解了它們,你會(huì)對(duì)JavaScript引擎如何實(shí)現(xiàn)異步事件有一個(gè)很好的認(rèn)識(shí)。這是一個(gè)一維的圖標(biāo):垂直方向表示時(shí)間,藍(lán)色的區(qū)塊表示JavaScript代碼執(zhí)行塊。例如第一個(gè)JavaScript代碼執(zhí)行塊需要大約18ms,鼠標(biāo)點(diǎn)擊所觸發(fā)的代碼執(zhí)行塊需要11ms等等。
由于JavaScript引擎同一時(shí)間只執(zhí)行一條代碼(這是由于JavaScript單線程的性質(zhì)),所以每一個(gè)JavaScript代碼執(zhí)行塊會(huì)“阻塞”其它異步事件的執(zhí)行。這就意味著當(dāng)一個(gè)異步事件發(fā)生(例如,鼠標(biāo)點(diǎn)擊,計(jì)時(shí)器被觸發(fā),或者Ajax異步請(qǐng)求)后,這些事件的回調(diào)函數(shù)將排在執(zhí)行隊(duì)列的最后等待執(zhí)行(實(shí)際上,排隊(duì)的方式根據(jù)瀏覽器的不同而不同,所以這里只是一個(gè)簡(jiǎn)化);
從第一個(gè)JavaScript執(zhí)行塊開始研究,在第一個(gè)執(zhí)行塊中兩個(gè)計(jì)時(shí)器被初始化:一個(gè)10ms的setTimeout()和一個(gè)10ms的setInterval()。依據(jù)何時(shí)何地計(jì)時(shí)器被初始化(計(jì)時(shí)器初始化完畢后就會(huì)開始計(jì)時(shí)),計(jì)時(shí)器實(shí)際上會(huì)在第一個(gè)代碼塊執(zhí)行完畢前被觸發(fā)。但是,計(jì)時(shí)器上綁定的函數(shù)不會(huì)立即執(zhí)行(不被立即執(zhí)行的原因是JavaScript是單線程的)。實(shí)際上,被延遲的函數(shù)將依次排在執(zhí)行隊(duì)列的最后,等待下一次恰當(dāng)?shù)臅r(shí)間再執(zhí)行。
此外,在第一個(gè)JavaScript執(zhí)行塊中我們看到了一個(gè)“鼠標(biāo)點(diǎn)擊”事件發(fā)生了。一個(gè)JavaScript回調(diào)函數(shù)綁定在這個(gè)異步事件上了(我們從來(lái)不知道用戶什么時(shí)候執(zhí)行這個(gè)(點(diǎn)擊)事件,因此認(rèn)為它是異步的),這個(gè)函數(shù)不會(huì)被立即執(zhí)行,和上面的計(jì)時(shí)器一樣,它將排在執(zhí)行隊(duì)列的最后,等待下一次恰當(dāng)?shù)臅r(shí)候執(zhí)行。
當(dāng)?shù)谝粋€(gè)JavaScript執(zhí)行塊執(zhí)行完畢后,瀏覽器會(huì)立即問一個(gè)問題:哪個(gè)函數(shù)(語(yǔ)句)在等待被執(zhí)行?在這時(shí),一個(gè)“鼠標(biāo)點(diǎn)擊事件處理函數(shù)”和一個(gè)“計(jì)時(shí)器回調(diào)函數(shù)”都在等待執(zhí)行。瀏覽器會(huì)選擇一個(gè)(實(shí)際上選擇了“鼠標(biāo)點(diǎn)擊事件的處理函數(shù)”,因?yàn)橛蓤D可知它是先進(jìn)隊(duì)的)立即執(zhí)行。而“計(jì)時(shí)器回調(diào)函數(shù)”將等待下次適合的時(shí)間執(zhí)行。
注意,當(dāng)“鼠標(biāo)點(diǎn)擊事件處理函數(shù)”執(zhí)行的時(shí)候,setInterval的回調(diào)函數(shù)第一次被觸發(fā)了。和setTimeout的回調(diào)函數(shù)一樣,它將排到執(zhí)行隊(duì)列的最后等待執(zhí)行。但是,一定要注意這一點(diǎn):當(dāng)setInterval回調(diào)函數(shù)第二次被觸發(fā)時(shí)(此時(shí)setTimeout函數(shù)仍在執(zhí)行)setTimeout的第一次觸發(fā)將被拋棄掉。當(dāng)一個(gè)很長(zhǎng)的代碼塊在執(zhí)行時(shí),可能把所有的setInterval回調(diào)函數(shù)都排在執(zhí)行隊(duì)列的后面,代碼塊執(zhí)行完之后,結(jié)果便會(huì)是一大串的setInterval回調(diào)函數(shù)等待執(zhí)行,并且這些函數(shù)之間沒有間隔,直到全部完成。所以,瀏覽器傾向于的當(dāng)沒有更多interval的處理函數(shù)在排隊(duì)時(shí)再將下一個(gè)處理函數(shù)排到隊(duì)尾(這是由于間隔的問題)。
我們能夠發(fā)現(xiàn),當(dāng)?shù)谌齻€(gè)setInterval回調(diào)函數(shù)被觸發(fā)時(shí),之前的setInterval回調(diào)函數(shù)仍在執(zhí)行。這就說(shuō)明了一個(gè)很重要的事實(shí):setInterval不會(huì)考慮當(dāng)前正在執(zhí)行什么,而把所有的堵塞的函數(shù)排到隊(duì)列尾部。這意味著兩次setInterval回調(diào)函數(shù)之間的時(shí)間間隔會(huì)被犧牲掉(縮減)。
最后,當(dāng)?shù)诙€(gè)setInterval回調(diào)函數(shù)執(zhí)行完畢后,我們可以看到?jīng)]有任何程序等待JavaScript引擎執(zhí)行了。這就意味著瀏覽器現(xiàn)在在等待一個(gè)新的異步事件的發(fā)生。在50ms時(shí)一個(gè)新的setInterval回調(diào)函數(shù)再次被觸發(fā),這時(shí),沒有任何的執(zhí)行塊阻塞它的執(zhí)行了。所以它會(huì)立刻被執(zhí)行。讓我們用一個(gè)例子來(lái)闡明setTimeout和setInterval之間的區(qū)別:
setTimeout(function(){
/* Some long block of code... */
setTimeout(arguments.callee, 10);
}, 10);
setInterval(function(){
/* Some long block of code... */
}, 10);
setTimeout(function(){
/* Some long block of code... */
setTimeout(arguments.callee, 10);
}, 10);
setInterval(function(){
/* Some long block of code... */
}, 10);
這兩句代碼乍一看沒什么差別,但是它們是不同的。setTimeout回調(diào)函數(shù)的執(zhí)行和上一次執(zhí)行之間的間隔至少有10ms(可能會(huì)更多,但不會(huì)少于10ms),而setInterval的回調(diào)函數(shù)將嘗試每隔10ms執(zhí)行一次,不論上次是否執(zhí)行完畢。
總結(jié)
◆JavaScript引擎是單線程的,強(qiáng)制所有的異步事件排隊(duì)等待執(zhí)行;
◆setTimeout和 setInterval在執(zhí)行異步代碼的時(shí)候有著根本的不同;
◆如果一個(gè)計(jì)時(shí)器被阻塞而不能立即執(zhí)行,它將延遲執(zhí)行直到下一次可能執(zhí)行的時(shí)間點(diǎn)才被執(zhí)行(比期望的時(shí)間間隔要長(zhǎng)些);
◆如果setInterval回調(diào)函數(shù)的執(zhí)行時(shí)間將足夠長(zhǎng)(比指定的時(shí)間間隔長(zhǎng)),它們將連續(xù)執(zhí)行并且彼此之間沒有時(shí)間間隔。
上述這些知識(shí)點(diǎn)都是非常重要的。了解了JavaScript引擎是如何工作的,尤其是大量的異步事件(連續(xù))發(fā)生時(shí),才能為構(gòu)建高級(jí)應(yīng)用程序打好基礎(chǔ)。
腳本之家推薦閱讀:雕蟲無(wú)小技 JavaScript初學(xué)者的10個(gè)迷你技巧
復(fù)制代碼 代碼如下:
var id = setTimeout(fn, delay);
初始化一個(gè)計(jì)時(shí)器,然后在指定的時(shí)間間隔后執(zhí)行。該函數(shù)返回一個(gè)唯一的標(biāo)志ID(Number類型),我們可以使用它來(lái)取消計(jì)時(shí)器。
復(fù)制代碼 代碼如下:
var id = setInterval(fn, delay);
和setTimeout有些類似,但它是連續(xù)調(diào)用一個(gè)函數(shù)(時(shí)間間隔是delay參數(shù))直到它被取消。
復(fù)制代碼 代碼如下:
clearInterval(id);, clearTimeout(id);
使用計(jì)時(shí)器ID(setTimeout 和 setInterval的返回值)來(lái)取消計(jì)時(shí)器回調(diào)的發(fā)生。
為了理解計(jì)時(shí)器的內(nèi)在執(zhí)行原理,有一個(gè)重要的概念需要加以探討:計(jì)時(shí)器的延遲(delay)是無(wú)法得到保障的。由于所有JavaScript代碼是在一個(gè)線程里執(zhí)行的,所有異步事件(例如,鼠標(biāo)點(diǎn)擊和計(jì)時(shí)器)只有擁有執(zhí)行機(jī)會(huì)時(shí)才會(huì)執(zhí)行。用一個(gè)很好的圖表加以說(shuō)明:

在這個(gè)圖表中有許多信息需要理解,如果完全理解了它們,你會(huì)對(duì)JavaScript引擎如何實(shí)現(xiàn)異步事件有一個(gè)很好的認(rèn)識(shí)。這是一個(gè)一維的圖標(biāo):垂直方向表示時(shí)間,藍(lán)色的區(qū)塊表示JavaScript代碼執(zhí)行塊。例如第一個(gè)JavaScript代碼執(zhí)行塊需要大約18ms,鼠標(biāo)點(diǎn)擊所觸發(fā)的代碼執(zhí)行塊需要11ms等等。
由于JavaScript引擎同一時(shí)間只執(zhí)行一條代碼(這是由于JavaScript單線程的性質(zhì)),所以每一個(gè)JavaScript代碼執(zhí)行塊會(huì)“阻塞”其它異步事件的執(zhí)行。這就意味著當(dāng)一個(gè)異步事件發(fā)生(例如,鼠標(biāo)點(diǎn)擊,計(jì)時(shí)器被觸發(fā),或者Ajax異步請(qǐng)求)后,這些事件的回調(diào)函數(shù)將排在執(zhí)行隊(duì)列的最后等待執(zhí)行(實(shí)際上,排隊(duì)的方式根據(jù)瀏覽器的不同而不同,所以這里只是一個(gè)簡(jiǎn)化);
從第一個(gè)JavaScript執(zhí)行塊開始研究,在第一個(gè)執(zhí)行塊中兩個(gè)計(jì)時(shí)器被初始化:一個(gè)10ms的setTimeout()和一個(gè)10ms的setInterval()。依據(jù)何時(shí)何地計(jì)時(shí)器被初始化(計(jì)時(shí)器初始化完畢后就會(huì)開始計(jì)時(shí)),計(jì)時(shí)器實(shí)際上會(huì)在第一個(gè)代碼塊執(zhí)行完畢前被觸發(fā)。但是,計(jì)時(shí)器上綁定的函數(shù)不會(huì)立即執(zhí)行(不被立即執(zhí)行的原因是JavaScript是單線程的)。實(shí)際上,被延遲的函數(shù)將依次排在執(zhí)行隊(duì)列的最后,等待下一次恰當(dāng)?shù)臅r(shí)間再執(zhí)行。
此外,在第一個(gè)JavaScript執(zhí)行塊中我們看到了一個(gè)“鼠標(biāo)點(diǎn)擊”事件發(fā)生了。一個(gè)JavaScript回調(diào)函數(shù)綁定在這個(gè)異步事件上了(我們從來(lái)不知道用戶什么時(shí)候執(zhí)行這個(gè)(點(diǎn)擊)事件,因此認(rèn)為它是異步的),這個(gè)函數(shù)不會(huì)被立即執(zhí)行,和上面的計(jì)時(shí)器一樣,它將排在執(zhí)行隊(duì)列的最后,等待下一次恰當(dāng)?shù)臅r(shí)候執(zhí)行。
當(dāng)?shù)谝粋€(gè)JavaScript執(zhí)行塊執(zhí)行完畢后,瀏覽器會(huì)立即問一個(gè)問題:哪個(gè)函數(shù)(語(yǔ)句)在等待被執(zhí)行?在這時(shí),一個(gè)“鼠標(biāo)點(diǎn)擊事件處理函數(shù)”和一個(gè)“計(jì)時(shí)器回調(diào)函數(shù)”都在等待執(zhí)行。瀏覽器會(huì)選擇一個(gè)(實(shí)際上選擇了“鼠標(biāo)點(diǎn)擊事件的處理函數(shù)”,因?yàn)橛蓤D可知它是先進(jìn)隊(duì)的)立即執(zhí)行。而“計(jì)時(shí)器回調(diào)函數(shù)”將等待下次適合的時(shí)間執(zhí)行。
注意,當(dāng)“鼠標(biāo)點(diǎn)擊事件處理函數(shù)”執(zhí)行的時(shí)候,setInterval的回調(diào)函數(shù)第一次被觸發(fā)了。和setTimeout的回調(diào)函數(shù)一樣,它將排到執(zhí)行隊(duì)列的最后等待執(zhí)行。但是,一定要注意這一點(diǎn):當(dāng)setInterval回調(diào)函數(shù)第二次被觸發(fā)時(shí)(此時(shí)setTimeout函數(shù)仍在執(zhí)行)setTimeout的第一次觸發(fā)將被拋棄掉。當(dāng)一個(gè)很長(zhǎng)的代碼塊在執(zhí)行時(shí),可能把所有的setInterval回調(diào)函數(shù)都排在執(zhí)行隊(duì)列的后面,代碼塊執(zhí)行完之后,結(jié)果便會(huì)是一大串的setInterval回調(diào)函數(shù)等待執(zhí)行,并且這些函數(shù)之間沒有間隔,直到全部完成。所以,瀏覽器傾向于的當(dāng)沒有更多interval的處理函數(shù)在排隊(duì)時(shí)再將下一個(gè)處理函數(shù)排到隊(duì)尾(這是由于間隔的問題)。
我們能夠發(fā)現(xiàn),當(dāng)?shù)谌齻€(gè)setInterval回調(diào)函數(shù)被觸發(fā)時(shí),之前的setInterval回調(diào)函數(shù)仍在執(zhí)行。這就說(shuō)明了一個(gè)很重要的事實(shí):setInterval不會(huì)考慮當(dāng)前正在執(zhí)行什么,而把所有的堵塞的函數(shù)排到隊(duì)列尾部。這意味著兩次setInterval回調(diào)函數(shù)之間的時(shí)間間隔會(huì)被犧牲掉(縮減)。
最后,當(dāng)?shù)诙€(gè)setInterval回調(diào)函數(shù)執(zhí)行完畢后,我們可以看到?jīng)]有任何程序等待JavaScript引擎執(zhí)行了。這就意味著瀏覽器現(xiàn)在在等待一個(gè)新的異步事件的發(fā)生。在50ms時(shí)一個(gè)新的setInterval回調(diào)函數(shù)再次被觸發(fā),這時(shí),沒有任何的執(zhí)行塊阻塞它的執(zhí)行了。所以它會(huì)立刻被執(zhí)行。讓我們用一個(gè)例子來(lái)闡明setTimeout和setInterval之間的區(qū)別:
復(fù)制代碼 代碼如下:
setTimeout(function(){
/* Some long block of code... */
setTimeout(arguments.callee, 10);
}, 10);
setInterval(function(){
/* Some long block of code... */
}, 10);
setTimeout(function(){
/* Some long block of code... */
setTimeout(arguments.callee, 10);
}, 10);
setInterval(function(){
/* Some long block of code... */
}, 10);
這兩句代碼乍一看沒什么差別,但是它們是不同的。setTimeout回調(diào)函數(shù)的執(zhí)行和上一次執(zhí)行之間的間隔至少有10ms(可能會(huì)更多,但不會(huì)少于10ms),而setInterval的回調(diào)函數(shù)將嘗試每隔10ms執(zhí)行一次,不論上次是否執(zhí)行完畢。
總結(jié)
◆JavaScript引擎是單線程的,強(qiáng)制所有的異步事件排隊(duì)等待執(zhí)行;
◆setTimeout和 setInterval在執(zhí)行異步代碼的時(shí)候有著根本的不同;
◆如果一個(gè)計(jì)時(shí)器被阻塞而不能立即執(zhí)行,它將延遲執(zhí)行直到下一次可能執(zhí)行的時(shí)間點(diǎn)才被執(zhí)行(比期望的時(shí)間間隔要長(zhǎng)些);
◆如果setInterval回調(diào)函數(shù)的執(zhí)行時(shí)間將足夠長(zhǎng)(比指定的時(shí)間間隔長(zhǎng)),它們將連續(xù)執(zhí)行并且彼此之間沒有時(shí)間間隔。
上述這些知識(shí)點(diǎn)都是非常重要的。了解了JavaScript引擎是如何工作的,尤其是大量的異步事件(連續(xù))發(fā)生時(shí),才能為構(gòu)建高級(jí)應(yīng)用程序打好基礎(chǔ)。
原文作者:John Resig
相關(guān)文章
BootstrapValidator不觸發(fā)校驗(yàn)的實(shí)現(xiàn)代碼
BootstrapValidator是基于bootstrap3的jquery表單驗(yàn)證插件,是最適合bootstrap框架的表單驗(yàn)證插件,本文給大家介紹BootstrapValidator不觸發(fā)校驗(yàn)的實(shí)現(xiàn)代碼,感興趣的朋友一起看看吧2016-09-09js實(shí)現(xiàn)精美的圖片跟隨鼠標(biāo)效果實(shí)例
這篇文章主要介紹了js實(shí)現(xiàn)精美的圖片跟隨鼠標(biāo)效果,實(shí)例分析了javascript鼠標(biāo)事件及頁(yè)面樣式的操作技巧,需要的朋友可以參考下2015-05-05uniapp自定義tabbar的方法(支持中間凸起、角標(biāo)、動(dòng)態(tài)隱藏tab和全端適用)
一個(gè)項(xiàng)目有多個(gè)角色,比如醫(yī)生和患者,tabBar跳轉(zhuǎn)的路徑不一樣,但是在pages.json中無(wú)法配置多個(gè)tabBar,這時(shí)候就要自定義tabBar了,下面這篇文章主要給大家介紹了關(guān)于uniapp自定義tabbar(支持中間凸起、角標(biāo)、動(dòng)態(tài)隱藏tab和全端適用)的相關(guān)資料,需要的朋友可以參考下2023-04-04