欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JavaScript函數(shù)節(jié)流和函數(shù)防抖之間的區(qū)別

 更新時間:2017年02月15日 08:27:36   作者:wall''''''''''''''''s  
本文主要介紹了JavaScript函數(shù)節(jié)流和函數(shù)防抖之間的區(qū)別。具有很好的參考價值,下面跟著小編一起來看下吧

一、概念解釋

函數(shù)節(jié)流和函數(shù)防抖,兩者都是優(yōu)化高頻率執(zhí)行js代碼的一種手段。

大家大概都知道舊款電視機的工作原理,就是一行行得掃描出色彩到屏幕上,然后組成一張張圖片。由于肉眼只能分辨出一定頻率的變化,當(dāng)高頻率的掃描,人類是感覺不出來的。反而形成一種視覺效果,就是一張圖。就像高速旋轉(zhuǎn)的風(fēng)扇,你看不到扇葉,只看到了一個圓一樣。

同理,可以類推到j(luò)s代碼。在一定時間內(nèi),代碼執(zhí)行的次數(shù)不一定要非常多。達到一定頻率就足夠了。因為跑得越多,帶來的效果也是一樣。倒不如,把js代碼的執(zhí)行次數(shù)控制在合理的范圍。既能節(jié)省瀏覽器CPU資源,又能讓頁面瀏覽更加順暢,不會因為js的執(zhí)行而發(fā)生卡頓。這就是函數(shù)節(jié)流和函數(shù)防抖要做的事。

函數(shù)節(jié)流是指一定時間內(nèi)js方法只跑一次。比如人的眨眼睛,就是一定時間內(nèi)眨一次。這是函數(shù)節(jié)流最形象的解釋。

函數(shù)防抖是指頻繁觸發(fā)的情況下,只有足夠的空閑時間,才執(zhí)行代碼一次。比如生活中的坐公交,就是一定時間內(nèi),如果有人陸續(xù)刷卡上車,司機就不會開車。只有別人沒刷卡了,司機才開車。

二、函數(shù)節(jié)流

函數(shù)節(jié)流應(yīng)用的實際場景,多數(shù)在監(jiān)聽頁面元素滾動事件的時候會用到。因為滾動事件,是一個高頻觸發(fā)的事件。以下是監(jiān)聽頁面元素滾動的示例代碼:

// 函數(shù)節(jié)流
var canRun = true;
document.getElementById("throttle").onscroll = function(){
 if(!canRun){
 // 判斷是否已空閑,如果在執(zhí)行中,則直接return
 return;
 }
 canRun = false;
 setTimeout(function(){
 console.log("函數(shù)節(jié)流");
 canRun = true;
 }, 300);
};

函數(shù)節(jié)流的要點是,聲明一個變量當(dāng)標志位,記錄當(dāng)前代碼是否在執(zhí)行。

如果空閑,則可以正常觸發(fā)方法執(zhí)行。

如果代碼正在執(zhí)行,則取消這次方法執(zhí)行,直接return。

這個方法的作用是監(jiān)聽ID為throttle元素的滾動事件。

當(dāng)canRun為true,則代表現(xiàn)在的滾動處理事件是空閑的,可以使用。

通過關(guān)卡if(!canRun),等于就拿到了通行證。然后下一步的操作就是立馬將關(guān)卡關(guān)上canRun=false。這樣,其他請求執(zhí)行滾動事件的方法,就被擋回去了。

接著用setTimeout規(guī)定最小的時間間隔300,接著再執(zhí)行setTimeout方法體里面的內(nèi)容。

最后,等setTimeout里面的方法都執(zhí)行完畢,才釋放關(guān)卡canRun=true,允許下一個訪問者進來。

這個函數(shù)節(jié)流的實現(xiàn)形式,需要注意的是執(zhí)行的間隔時間是>=300ms。如果具體執(zhí)行的方法是包含callback的,也可以將canRun=true這一步放到callback中。理解了函數(shù)節(jié)流的關(guān)卡設(shè)置重點,其實改起來就簡單多了。

三、函數(shù)防抖

函數(shù)防抖的應(yīng)用場景,最常見的就是用戶注冊時候的手機號碼驗證和郵箱驗證了。只有等用戶輸入完畢后,前端才需要檢查格式是否正確,如果不正確,再彈出提示語。以下還是以頁面元素滾動監(jiān)聽的例子,來進行解析:

// 函數(shù)防抖
var timer = false;
document.getElementById("debounce").onscroll = function(){
 clearTimeout(timer); // 清除未執(zhí)行的代碼,重置回初始化狀態(tài)
 timer = setTimeout(function(){
 console.log("函數(shù)防抖");
 }, 300);
}; 

函數(shù)節(jié)流的要點,也是需要一個setTimeout來輔助實現(xiàn)。延遲執(zhí)行需要跑的代碼。

如果方法多次觸發(fā),則把上次記錄的延遲執(zhí)行代碼用clearTimeout清掉,重新開始。

如果計時完畢,沒有方法進來訪問觸發(fā),則執(zhí)行代碼。

這個方法的作用是監(jiān)聽ID為debounce元素的滾動事件

進入滾動事件方法體的時候,做的第一件事就是清除上次未執(zhí)行的setTimeout。而setTimeout的引用id由變量timer記錄。

clearTimeout方法,允許傳入無效的值。所以這里直接執(zhí)行clearTimeout即可。

然后,將需要執(zhí)行的代碼放入setTimeout中,再返回setTimeout引用給timer緩存。

如果倒計時300ms以后,還沒有新的方法觸發(fā)滾動事件,則執(zhí)行setTimeout中的代碼。

函數(shù)防抖的實現(xiàn)重點,就是巧用setTimeout做緩存池,而且可以輕易地清除待執(zhí)行的代碼。

其實,用隊列的方式也可以做到這種效果。這里就不深入了。

四、在線demo

這是我寫的一個測試demo,把鼠標移動到模塊上方,滾動滾輪,即可在控制臺查看輸出效果。

demo地址:https://wall-wxk.github.io/blogDemo/2017/02/15/throttleAndDebounce.html

以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時也希望多多支持腳本之家!

相關(guān)文章

最新評論