zoukankan      html  css  js  c++  java
  • 函数防抖与函数节流 封装好的debounce和throttle函数

    /**
     * 空闲控制 返回函数连续调用时,空闲时间必须大于或等于 wait,func 才会执行
     *
     * @param  {function} func        传入函数,最后一个参数是额外增加的this对象,.apply(this, args) 这种方式,this无法传递进函数
     * @param  {number}   wait        表示时间窗口的间隔
     * @param  {boolean}  immediate   设置为ture时,调用触发于开始边界而不是结束边界
     * @return {function}             返回客户调用函数
     */
    const debounce = function(func, wait, immediate) {
        let timeout, args, context, timestamp, result;
    
        const later = function() {
            // 据上一次触发时间间隔
            let last = Number(new Date()) - timestamp;
    
            // 上次被包装函数被调用时间间隔last小于设定时间间隔wait
            if (last < wait && last > 0) {
                timeout = setTimeout(later, wait - last);
            } else {
                timeout = null;
                // 如果设定为immediate===true,因为开始边界已经调用过了此处无需调用
                if (!immediate) {
                    result = func.call(context, ...args, context);
                    if (!timeout) {
                        context = args = null;
                    }
                }
            }
        };
    
        return function(..._args) {
            context = this;
            args = _args;
            timestamp = Number(new Date());
            const callNow = immediate && !timeout;
            // 如果延时不存在,重新设定延时
            if (!timeout) {
                timeout = setTimeout(later, wait);
            }
            if (callNow) {
                result = func.call(context, ...args, context);
                context = args = null;
            }
    
            return result;
        };
    };
    /**
     * 频率控制 返回函数连续调用时,func 执行频率限定为 次 / wait
     *
     * @param  {function}   func      传入函数
     * @param  {number}     wait      表示时间窗口的间隔
     * @param  {object}     options   如果想忽略开始边界上的调用,传入{leading: false}。
     *                                如果想忽略结尾边界上的调用,传入{trailing: false}
     * @return {function}             返回客户调用函数
     */
    const throttle = function(func, wait, options) {
        let context, args, result;
        let timeout = null;
        // 上次执行时间点
        let previous = 0;
        if (!options) options = {};
        // 延迟执行函数
        let later = function() {
            // 若设定了开始边界不执行选项,上次执行时间始终为0
            previous = options.leading === false ? 0 : Number(new Date());
            timeout = null;
            result = func.apply(context, args);
            if (!timeout) context = args = null;
        };
        return function(..._args) {
            let now = Number(new Date());
            // 首次执行时,如果设定了开始边界不执行选项,将上次执行时间设定为当前时间。
            if (!previous && options.leading === false) previous = now;
            // 延迟执行时间间隔
            let remaining = wait - (now - previous);
            context = this;
            args = _args;
            // 延迟时间间隔remaining小于等于0,表示上次执行至此所间隔时间已经超过一个时间窗口
            // remaining大于时间窗口wait,表示客户端系统时间被调整过
            if (remaining <= 0 || remaining > wait) {
                clearTimeout(timeout);
                timeout = null;
                previous = now;
                result = func.apply(context, args);
                if (!timeout) context = args = null;
                //如果延迟执行不存在,且没有设定结尾边界不执行选项
            } else if (!timeout && options.trailing !== false) {
                timeout = setTimeout(later, remaining);
            }
            return result;
        };
    };
  • 相关阅读:
    Heap(堆)和stack(栈)有的区别是什么。
    i++和++i的深入理解
    JDBC之java数据库的连接与简单的sql语句执行
    java前三本基础知识总结
    数据库的一些基础
    SQL 同时查看2个表
    JMeter 问题
    Linux 常用命令
    java io (一)
    验证密码必须是字母加数字的组合
  • 原文地址:https://www.cnblogs.com/chenzeyongjsj/p/10394042.html
Copyright © 2011-2022 走看看