詳解javascript高級定時器
setTimeout()和setInterval()可以用來創(chuàng)建定時器,其基本的用法這里就不再做介紹了。這里主要介紹一下javascript的代碼隊(duì)列。在javascript中沒有任何代碼是立即執(zhí)行的,一旦進(jìn)程空閑則盡快執(zhí)行。所以說定時器中設(shè)置的時間并不代表執(zhí)行時間就一定相符,而是代表代碼會在指定時間間隔后加入到隊(duì)列中進(jìn)行等待。如果在這個時間點(diǎn)上,隊(duì)列中沒有其他東西,那么這段代碼就會被執(zhí)行,表面上看上去好像代碼就在精確指定的時間點(diǎn)上執(zhí)行了。所以就會產(chǎn)生一些問題。
重復(fù)定時器
通常,我們使用setInterval方法來以相同時間間隔重復(fù)執(zhí)行某段代碼。但是使用該方法會有兩個問題:第一個就是某些間隔會被跳過;第二個就是多個定時器的代碼執(zhí)行之間的間隔可能會比預(yù)期的小。
在這里,我們來舉個例子:如果某個onclick事件處理程序使用setInterval設(shè)置了一個200ms間隔的重復(fù)定時器,如果事件處理程序花了300ms的時間完成,就會跳過一個時間間隔同時運(yùn)行著一個定時器代碼。
我們也可以通過下面的代碼來得到結(jié)論:
//重復(fù)定時器
var i =0;
setInterval(function(){
//如果事件處理時間長于間隔時間
i++;
for(var j=0;j<100000000;j++){}
document.write(i+' ');
},100);
//可以明顯感覺到時間間隔不相等
為了避免這種時間間隔的問題,我們可以采用鏈?zhǔn)秸{(diào)用setTimeout方法來取代setInterval。
//可以采用鏈?zhǔn)秸{(diào)用setTimeout來取代setInterval
var i = 0;
setTimeout(function(){
//處理內(nèi)容
i++;
for(var j=0;j<100000000;j++){}
document.write(i+' ');
//
setTimeout(arguments.callee,100);
},100);
//這樣處理效果明顯好多了。
每次函數(shù)執(zhí)行的時候都會創(chuàng)建一個新的定時器,第二個setTimeout調(diào)用使用了arguments.callee來獲取對當(dāng)前執(zhí)行的函數(shù)的引用,并為其設(shè)置另外一個定時器。這樣做是為了在前一個定時器代碼執(zhí)行完之前,不會向隊(duì)列插入新的定時器代碼,確保不會有任何缺失的間隔,也保證了在下一次定時器代碼執(zhí)行之前,至少要等待指定的間隔,避免了連續(xù)的運(yùn)行。可謂一舉兩得,現(xiàn)在主流框架中的動畫一般都是這樣來實(shí)現(xiàn)重復(fù)定時的。
函數(shù)節(jié)流
定時器不僅僅是用來定時的,也可以用來緩解瀏覽器的壓力。瀏覽器中某些計(jì)算和處理要比其他的昂貴很多,比如說DOM操作,就會需要更多的內(nèi)存和CPU時間,連續(xù)使用過多的DOM操作可能會導(dǎo)致瀏覽器掛起,甚至崩潰。
函數(shù)節(jié)流的基本思想就是,某些代碼不可以在沒有間斷的情況連續(xù)重復(fù)執(zhí)行。第一次調(diào)用函數(shù),創(chuàng)建一個定時器,在指定的時間間隔之后運(yùn)行代碼。當(dāng)?shù)诙握{(diào)用該函數(shù)時,它會清除前一次的定時器并設(shè)置一個。目的就是為了在執(zhí)行函數(shù)的請求停止一段時間后再執(zhí)行。
代碼如下:
//再來談?wù)労瘮?shù)節(jié)流
function throttle(method,context){
clearTimeout(method.tId);
method.tId = setTimeout(function(){
method.call(context);
},100);
}
//該函數(shù)接受兩個參數(shù),第一個是要執(zhí)行的函數(shù),第二個是作用域。
//使用方法demo
//未使用情況:
window.onresize = function(){
var div = document.getElementByTagName(body);
div.style.height = div.offsetWidth +'px';
}
//使用情況;
function resizeDiv(){
var div = document.getElementByTagName(body);
div.style.height = div.offsetWidth +'px';
}
window.onresize = function(){
throttle(resizeDiv);
};
//只要代碼是周期性執(zhí)行的,都應(yīng)該使用節(jié)流。
這樣給用戶的感覺并不會很大,確是給瀏覽器減少了不少的壓力,函數(shù)節(jié)流也是很多框架常用的技巧之一。
以上就是關(guān)于javascript高級定時器的相關(guān)介紹,希望對大家的學(xué)習(xí)有所幫助。
相關(guān)文章
基于JavaScript實(shí)現(xiàn)一個簡單的事件觸發(fā)器
這篇文章主要為大家詳細(xì)介紹了如何使用JavaScript實(shí)現(xiàn)一個簡單的事件觸發(fā)器,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-01-01
深入理解requireJS-實(shí)現(xiàn)一個簡單的模塊加載器
本篇文章主要介紹了深入理解requireJS-實(shí)現(xiàn)一個簡單的模塊加載器,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-01-01
Bootstrap Table 在指定列中添加下拉框控件并獲取所選值
通過 bootstrap-table 的Column 配置項(xiàng)中的formatter,將獲取到的數(shù)據(jù)轉(zhuǎn)換為包含數(shù)據(jù)的 select 控件。然后根據(jù)用戶選擇項(xiàng)更新對應(yīng)單元格數(shù)據(jù),最后通過getallselection方法獲取所選行數(shù)據(jù)2017-07-07
前端實(shí)現(xiàn)文本超出指定行數(shù)顯示"展開"和"收起"效果詳細(xì)步驟
本文介紹如何使用JavaScript原生代碼實(shí)現(xiàn)文本折疊展開效果,并提供方法指導(dǎo)如何在Vue或React等框架中修改實(shí)現(xiàn),詳細(xì)介紹了創(chuàng)建整體框架、設(shè)置樣式及利用JS控制元素的步驟,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-10-10
Bootstrap的基本應(yīng)用要點(diǎn)淺析
BootStrap是基于HTML、CSS和JavaScript的框架,使你只需要寫簡單的代碼就可以很快的搭建一個還不錯的前端框架,他是后端程序員的福音,使他們只需要專注業(yè)務(wù)邏輯,而無須浪費(fèi)太多的精力在界面設(shè)計(jì)上2016-12-12
JavaScript如何將base64圖片轉(zhuǎn)化為URL格式
這篇文章主要給大家介紹了關(guān)于JavaScript如何將base64圖片轉(zhuǎn)化為URL格式的相關(guān)資料,Base64是一種編碼方式,而不是真正的加密方式,即使算Base64也僅用作一個簡單的加密來保護(hù)某些數(shù)據(jù),而真正的加密通常都比較繁瑣,需要的朋友可以參考下2023-07-07
JavaScript+Html5實(shí)現(xiàn)按鈕復(fù)制文字到剪切板功能(手機(jī)網(wǎng)頁兼容)
在學(xué)習(xí)javascript的過程中,遇到一個問題就是基于JavaScript+Html5實(shí)現(xiàn)按鈕復(fù)制文字到剪切板功能,下面小編給大家分享下我的實(shí)現(xiàn)思路,感興趣的朋友可以參考下2017-03-03
JQuery入門——用one()方法綁定事件處理函數(shù)(僅觸發(fā)一次)
one()方法功能是為所選的元素綁定一個僅觸發(fā)一次的處理函數(shù),感興趣的朋友可以了解下它的調(diào)用語法為:one(type, [data], fn),閱讀本文或許有意外的收獲呢2013-02-02
javascript實(shí)現(xiàn)自由編輯圖片代碼詳解
這篇文章主要介紹了javascript實(shí)現(xiàn)自由編輯圖片代碼詳解,在當(dāng)下的的前端項(xiàng)目中,圖片功能可以說是非常常見的,圖片的展示、圖片的裁剪編輯、圖片的上傳等,那么我們的項(xiàng)目便來了個需求。,需要的朋友可以參考下2019-06-06

