Debounce and throttle are techniques used to prevent JavaScript code that runs in response to user actions from blocking the user interface (or otherwise running more than is desirable). They are similar but subtly different and can be confusing because they combine some of the more tricky aspects of working with JavaScript: manipulating context; passing functions around as first-class values and managing timers.

Because JavaScript is, by nature, single-threaded (only one piece of JavaScript code can execute at a time), timers provide a way to dance around this restriction, resulting in a rather oblique way of executing code. Secrets of the JavaScript Ninja

Debounce and when to use it

I’ve always found the term ‘debounce’ confusing so I did some research and found it comes from an electronics problem with mechanical switches, the electrical contacts of which can often make and break contact several times in a single push. This is commonly known as ‘switch bounce’.

Switch bouncing can cause problems. Imagine the situation where pressing a button toggles an LED on and off. If there are an even number of bounces, then the LED will toggle on and then immediately off again, giving the impression that nothing happened. Practical Electronics for Inventors, 4th Edition

The solution to the bounce effect is to ensure that only a single signal will be acted upon. Doing so is called ‘debouncing’ and it’s very similar to the situation we face in the browser when responding to pointing device, scroll and even keyboard events. Here’s a debounce method written in JavaScript.

/**
 *
 * Invoke a given callback after debounce function
 * hasn't been called for a specified number of milliseconds
 *
 * usage:
 * debounce( call_back, 500 )( ..arg );
 *
 * @param {Function} call_back
 * @param {Number} wait
 * @param {Object} this_argument
 **/

function debounce(call_back, wait, this_argument) {
    var timer = null;

    return function (...args) {
        var context = this_argument || this;
        window.clearTimeout(timer);
        timer = window.setTimeout(() => {
            timer = null;
            call_back.apply(context, args);
        }, wait);
    };
}

window.addEventListener("scroll", debounce(() =>{
    console.log("I am being debounced!");
}, 200));

In this example, the debounce() function acts as a proxy for the intended event handler to an arrow function that logs a message to the console to and serves to ensure that the handler is called once and after an event stream (by which I mean any number of events within a specified period) has concluded.

Throttle and when to use it

Throttling is slightly different because to as its name suggests to it limits the number of times a function can be called within a specified period. Here’s an example of throttling.

/**
 * Invoke a given callback immediately and no more frequently than
 * every `wait` ms until this function stops being called
 *
 * usage:
 * throttle( call_back, 500 )( ..arg );
 *
 * @param {Function} call_back
 * @param {Number} wait
 * @param {Object} this_argument
 */
function throttle(call_back, wait, this_argument) {

    var last_call_timestamp, timer;

    return function (...args) {
        var context = this_argument || this,
            now = +new Date();

        if (!last_call_timestamp || now >= last_call_timestamp + wait {
            last_call_timestamp = now;
            return call_back.apply(context, args);
        }

        clearTimeout(timer);

        timer = setTimeout(() =>{
            last_call_timestamp = now;
            call_back.apply(context, args);
        }, wait);
    };
}

window.addEventListener("scroll", throttle(() =>{

    console.log("I am being throttled!");

}, 200));

As with the previous example, this throttle() function is a proxy for a simple logging function but the behaviour is different. With throttle() the proxied function will be called immediately and at the specified intervals, as well as once after the event stream has concluded. The throttle function will be most useful when you need your code to respond immediately and continually but at intervals you specify.

I really hope this is useful to someone (most likely my future self 😆)