如何理解JS函數(shù)防抖和函數(shù)節(jié)流
概述
函數(shù)防抖和函數(shù)節(jié)流都是定義一個函數(shù),該函數(shù)接收一個函數(shù)作為參數(shù),并返回一個添加了防抖或節(jié)流功能后的函數(shù)。
因此可以將函數(shù)防抖和函數(shù)節(jié)流看作是一個函數(shù)工廠,負(fù)責(zé)對傳進(jìn)來的函數(shù)進(jìn)行相應(yīng)的加工改造,然后產(chǎn)出一個新的帶有某種功能的函數(shù)。
函數(shù)防抖是某一時間內(nèi)只執(zhí)行一次,而函數(shù)節(jié)流是間隔時間執(zhí)行
假如有這樣一個場景:在某一頁面,有一個按鈕是 "加載更多",這個按鈕的作用就是使用 ajax 從后端服務(wù)器請求更多的數(shù)據(jù)展示在頁面,我們都知道,ajax 請求的響應(yīng)是一個異步的,會存在一定的響應(yīng)時間,如果用戶在點擊了該按鈕后,立馬又點擊了一下該按鈕,按照常規(guī)回調(diào)函數(shù)的操作,回調(diào)函數(shù)會立馬又執(zhí)行一次,那么就是用戶短時間內(nèi)點擊了兩次 "加載更多" 按鈕后就會執(zhí)行回調(diào)函數(shù)向后端發(fā)起了兩次一模一樣的 ajax 請求,服務(wù)器會一一接收請求并處理返回數(shù)據(jù),短時間內(nèi)兩次請求還好,如果用戶連續(xù)點擊了 n 次 "加載更多" 按鈕呢?那就會短時間內(nèi)向服務(wù)器發(fā)送了 n 次一模一樣的請求,每次后端處理 ajax 請求后返回數(shù)據(jù)后頁面就會重新再渲染一次,盡管內(nèi)容沒有改變,這樣就會帶來性能上的問題,不僅為服務(wù)器帶去了壓力,也為瀏覽器帶來了不必要的渲染,這就是函數(shù)頻繁執(zhí)行帶來的副作用。
那么如何設(shè)置這個按鈕的點擊事件回調(diào)函數(shù)才能減少該函數(shù)頻繁執(zhí)行帶來的副作用呢?
1. 函數(shù)防抖(debounce)
函數(shù)防抖的設(shè)計思路就是在函數(shù)執(zhí)行前加一個等待時間,在這個等待時間內(nèi)如果該函數(shù)又需要執(zhí)行一次,就重新計算等待時間,再次等待,依次類推,直到等待時間到了,還沒有等到函數(shù)又需要執(zhí)行的情況,才會執(zhí)行這個函數(shù)。
就拿上述的場景來說,假設(shè)向后臺發(fā)送 ajax 請求的響應(yīng)時間大概是在 2s 左右,那就設(shè)置這個等待時間為 2s,當(dāng)用戶第一次點擊"加載更多"按鈕后,該回調(diào)函數(shù)并沒有立即執(zhí)行,也就是還沒有開始發(fā)送 ajax,此時該函數(shù)在等待,如果在 2s 內(nèi)用戶又點擊了"加載更多",那就重新計算等待時間,再等待 2s ,此時 2s 過去了,用戶沒再第三次點擊"加載更多"按鈕了,那么該函數(shù)就開始執(zhí)行了,向后端發(fā)送 ajax 請求了。
函數(shù)防抖實現(xiàn)方式如下:
function debounce(fn, delay){
let timeId = null
return function(){
let context = this
if(timeId){window.clearTimeout(timeId)}
timeId = setTimeout(()=>{
fn.apply(context, arguments)
timeId = null
},delay)
}
}
2. 函數(shù)節(jié)流(throttle)
函數(shù)節(jié)流的設(shè)計思路是在函數(shù)執(zhí)行后加一個冷卻時間,函數(shù)在第一次執(zhí)行時是立馬執(zhí)行,但在其執(zhí)行完后設(shè)置一個冷卻時間,在冷卻時間內(nèi),該函數(shù)不能再次執(zhí)行,直到冷卻時間結(jié)束允許該函數(shù)執(zhí)行了,才可以再次執(zhí)行。
就拿上述的場景來說,冷卻時間假設(shè)也設(shè)置為 2s,在用戶第一次點擊"加載更多" 后按鈕的回調(diào)函數(shù)就會執(zhí)行,也就是會向后臺發(fā)送 ajax 請求,此時用戶又立馬再次點擊了"加載更多"按鈕,由于此刻 2s 的冷卻時間還沒有到,那么就會給第二次的函數(shù)執(zhí)行加上一個延遲執(zhí)行。
function throttle(fn, delay){
let canUse = true
return function(){
if(canUse){
fn.apply(this, arguments)
canUse = false
setTimeout(()=>canUse=true, delay)
}
}
}
以上就是如何理解JS函數(shù)防抖和函數(shù)節(jié)流的詳細(xì)內(nèi)容,更多關(guān)于JS函數(shù)防抖和函數(shù)節(jié)流的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
javascript數(shù)字?jǐn)?shù)組去重復(fù)項的實現(xiàn)代碼
console.log 不支持ie,下面的代碼需要在火狐中測試,不然會有問題。2010-12-12
JavaScript語法高亮插件highlight.js用法詳解【附highlight.js本站下載】
這篇文章主要介紹了JavaScript語法高亮庫highlight.js用法,詳細(xì)分析了highlight.js的下載、調(diào)用及具體使用技巧,需要的朋友可以參考下2016-11-11
JavaScript事件學(xué)習(xí)小結(jié)(三)js事件對象
這篇文章主要介紹了JavaScript事件學(xué)習(xí)小結(jié)(三)js事件對象的相關(guān)資料,非常不錯具有參考借鑒價值,需要的朋友可以參考下2016-06-06
基于Bootstrap的標(biāo)簽頁組件及bootstrap-tab使用說明
這篇文章主要介紹了基于Bootstrap的標(biāo)簽頁組件及bootstrap-tab使用說明,需要的朋友可以參考下2017-07-07
javascript基于原型鏈的繼承及call和apply函數(shù)用法分析
這篇文章主要介紹了javascript基于原型鏈的繼承及call和apply函數(shù)用法,結(jié)合實例形式較為詳細(xì)的分析了javascript中繼承的概念、創(chuàng)建方法以及call和apply函數(shù)的功能與使用技巧,需要的朋友可以參考下2016-12-12
基于JavaScript實現(xiàn)TAB標(biāo)簽效果
js實現(xiàn)tab標(biāo)簽效果,在項目中經(jīng)常用到,今天抽點時間把我寫的js實現(xiàn)tab標(biāo)簽效果源碼分享給大家,對js tab標(biāo)簽代碼需要的朋友參考下2016-01-01
javascript瀑布流式圖片懶加載實例解析與優(yōu)化
這篇文章主要針對javascript瀑布流式圖片懶加載實例進(jìn)行解析與優(yōu)化,感興趣的小伙伴們可以參考一下2016-02-02

