函數(shù)節(jié)流與函數(shù)防抖

我們在一個項目中按鈕連續(xù)多次點擊栋豫,會造成一些問題挤安,比如連點按鈕會造成多次push頁面,或者造成多次網(wǎng)絡(luò)請求丧鸯。那么就需要寫一個防連點來解決這個問題蛤铜。

小程序button組件有一個disabled屬性,設(shè)置按鈕是否禁用的屬性骡送。
起初我想用這一屬性來解決按鈕連點的問題昂羡,但是事實證明真機上并木有什么卵用,還是會有連點的問題發(fā)生摔踱。。后來發(fā)現(xiàn)了函數(shù)節(jié)流和函數(shù)防抖怨愤,到此解決按鈕連點問題派敷。

什么是函數(shù)節(jié)流與函數(shù)防抖

函數(shù)節(jié)流(throttle): 指定時間間隔內(nèi)只會執(zhí)行一次任務(wù)。
函數(shù)防抖(debounce): 任務(wù)頻繁觸發(fā)的情況下撰洗,只有任務(wù)觸發(fā)的間隔超過指定間隔的時候篮愉,任務(wù)才會執(zhí)行。

  1. 函數(shù)節(jié)流例子:
    以判斷頁面是否滾動到底部為例差导,一般就是監(jiān)聽 window 對象的 scroll 事件试躏,然后函數(shù)體中寫判斷是否滾動到底部的邏輯:
$(window).scroll(function() {
    // 判斷是否滾動到底部的邏輯
    const pageHeight = $('body').height();
    const scrollTop = $(window).scrollTop();
    const winHeight = $(window).height();
    const sTop = pageHeight - scrollTop - winHeight;
    if (sTop > -100 && sTop <= 20) {
        console.log('end');
    }
});

這樣做的一個缺點就是比較消耗性能,因為當(dāng)在滾動的時候设褐,瀏覽器會無時不刻地在計算判斷是否滾動到底部的邏輯颠蕴,而在實際的場景中是不需要這么做的,在實際場景中可能是這樣的:在滾動過程中助析,每隔一段時間在去計算這個判斷邏輯犀被。而函數(shù)節(jié)流所做的工作就是每隔一段時間去執(zhí)行一次原本需要無時不刻地在執(zhí)行的函數(shù),所以在滾動事件中引入函數(shù)的節(jié)流是一個非常好的實踐:

$(window).scroll(throttle(function() {
    // 判斷是否滾動到底部的邏輯
    const pageHeight = $('body').height();
    const scrollTop = $(window).scrollTop();
    const winHeight = $(window).height();
    const sTop = pageHeight - scrollTop - winHeight;
    if (sTop > -100 && sTop <= 20) {
        console.log('end');
    }
}), 300)
function throttle(fn, gapTime) {
    let _lastTime = null;
    return function () {
        let _nowTime = + new Date()
        if (_nowTime - _lastTime > gapTime || !_lastTime) {
            fn();
             _lastTime = _nowTime
        }
    }
}

加上函數(shù)節(jié)流之后外冀,當(dāng)頁面再滾動的時候寡键,每隔 300ms 才會去執(zhí)行一次判斷邏輯。

  1. 函數(shù)防抖例子:
    以用戶注冊時驗證用戶名是否被占用為例雪隧,如今很多網(wǎng)站為了提高用戶體驗西轩,不會再輸入框失去焦點的時候再去判斷用戶名是否被占用员舵,而是在輸入的時候就在判斷這個用戶名是否已被注冊:
$('.user-name').on('input', function () {
    $.ajax({
        url: '/check',
        method: 'post',
        data: {
            username: $(this).val(),
        },
        success(data) {
            if (data.isRegistered) {
                $('.tips').text('該用戶名已被注冊!');
            } else {
                $('.tips').text('恭喜藕畔!該用戶名還未被注冊固灵!');
            }
        },
        error(error) {
            console.log(error);
        },
    });
});

很明顯,這樣的做法不好的是當(dāng)用戶輸入第一個字符的時候劫流,就開始請求判斷了巫玻,不僅對服務(wù)器的壓力增大了,對用戶體驗也未必比原來的好祠汇。而理想的做法應(yīng)該是這樣的仍秤,當(dāng)用戶輸入第一個字符后的一段時間內(nèi)如果還有字符輸入的話,那就暫時不去請求判斷用戶名是否被占用可很。在這里引入函數(shù)防抖就能很好地解決這個問題:

$('.user-name').on('input', debounce(function () {
    $.ajax({
        url: '/check',
        method: 'post',
        data: {
            username: $(this).val(),
        },
        success(data) {
            if (data.isRegistered) {
                $('.tips').text('該用戶名已被注冊诗力!');
            } else {
                $('.tips').text('恭喜!該用戶名還未被注冊我抠!');
            }
        },
        error(error) {
            console.log(error);
        },
    });
}), 300);
function debounce(fn, interval) {
    let timeout = null;
    return function () {
        clearTimeout(timeout);
        timeout = setTimeout(() => {
            fn.apply(this, arguments);
        }, interval);
    };
}

通過閉包保存一個標記來保存 setTimeout 返回的值苇本,每當(dāng)用戶輸入的時候把前一個 setTimeout clear 掉,然后又創(chuàng)建一個新的 setTimeout菜拓,這樣就能保證輸入字符后的 interval 間隔內(nèi)如果還有字符輸入的話瓣窄,就不會執(zhí)行 fn 函數(shù)了。

函數(shù)防抖和函數(shù)節(jié)流是在時間軸上控制函數(shù)的執(zhí)行次數(shù)纳鼎。
使用函數(shù)節(jié)流與函數(shù)防抖可以節(jié)約計算機資源

本文部分內(nèi)容從其他技術(shù)分享中學(xué)習(xí)俺夕,希望對你有所幫助
拓展閱讀:https://justclear.github.io/throttle-and-debounce/

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市贱鄙,隨后出現(xiàn)的幾起案子劝贸,更是在濱河造成了極大的恐慌,老刑警劉巖逗宁,帶你破解...
    沈念sama閱讀 218,204評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件纵柿,死亡現(xiàn)場離奇詭異趟薄,居然都是意外死亡魁兼,警方通過查閱死者的電腦和手機何恶,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來言缤,“玉大人嚼蚀,你說我怎么就攤上這事」苄” “怎么了轿曙?”我有些...
    開封第一講書人閱讀 164,548評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我导帝,道長守谓,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,657評論 1 293
  • 正文 為了忘掉前任您单,我火速辦了婚禮斋荞,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘虐秦。我一直安慰自己平酿,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,689評論 6 392
  • 文/花漫 我一把揭開白布悦陋。 她就那樣靜靜地躺著蜈彼,像睡著了一般。 火紅的嫁衣襯著肌膚如雪俺驶。 梳的紋絲不亂的頭發(fā)上幸逆,一...
    開封第一講書人閱讀 51,554評論 1 305
  • 那天,我揣著相機與錄音暮现,去河邊找鬼还绘。 笑死,一個胖子當(dāng)著我的面吹牛栖袋,可吹牛的內(nèi)容都是我干的拍顷。 我是一名探鬼主播,決...
    沈念sama閱讀 40,302評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼栋荸,長吁一口氣:“原來是場噩夢啊……” “哼菇怀!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起晌块,我...
    開封第一講書人閱讀 39,216評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎帅霜,沒想到半個月后匆背,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,661評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡身冀,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,851評論 3 336
  • 正文 我和宋清朗相戀三年钝尸,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片搂根。...
    茶點故事閱讀 39,977評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡珍促,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出剩愧,到底是詐尸還是另有隱情猪叙,我是刑警寧澤,帶...
    沈念sama閱讀 35,697評論 5 347
  • 正文 年R本政府宣布,位于F島的核電站穴翩,受9級特大地震影響犬第,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜芒帕,卻給世界環(huán)境...
    茶點故事閱讀 41,306評論 3 330
  • 文/蒙蒙 一歉嗓、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧背蟆,春花似錦鉴分、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至本砰,卻和暖如春碴裙,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背点额。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評論 1 270
  • 我被黑心中介騙來泰國打工舔株, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人还棱。 一個月前我還...
    沈念sama閱讀 48,138評論 3 370
  • 正文 我出身青樓载慈,卻偏偏與公主長得像,于是被迫代替她去往敵國和親珍手。 傳聞我的和親對象是個殘疾皇子办铡,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,927評論 2 355