节流防抖的使用_监听滚动节流

节流防抖的使用_监听滚动节流节流防抖的使用

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全家桶1年46,售后保障稳定

什么是防抖

当事件被触发后,延迟几秒后再执行回调,如果在这几秒内事件又被触发,则重新计时。如:游戏中的回城机制,中途打断后必须要重新回城,重新读条。

应用场景

用户在输入框中连续输入一串字符时,可以通过防抖策略,只有在输入完后,才执行查询的请求,这样可以有效减少次数,节约请求资源。
例如:实现输入框的防抖

//模拟ajax请求
function ajax(content) { 
   
  console.log('ajax request ' + content);
}

let inputa = document.getElementById('unDebounce');

inputa.addEventListener('keyup', function (e) { 
   
    ajax(e.target.value);
})

Jetbrains全家桶1年46,售后保障稳定

只要按下键盘,就会触发ajax请求。从资源上来说是很浪费的行为,实际应用中,用户是输出完整的字符后才会请求。可以进行优化:

function ajax(content) { 
   
  console.log('ajax request ' + content);
}

function debounce(fun, delay) { 
   
    return function (args) { 
   
        let that = this;
        let _args = args;
        clearTimeout(fun.id);
        fun.id = setTimeout(function () { 
   
            fun.call(that, _args);
        }, delay)
    }
}
    
let inputb = document.getElementById('debounce')

let debounceAjax = debounce(ajax, 500)

inputb.addEventListener('keyup', function (e) { 
   
        debounceAjax(e.target.value)
    })

加入了防抖后,在频繁的输入时不会发送请求,只有当在指定间隔内没有输入时,才会执行函数。如果停止输入但是在指定间隔内又输入,会重新触发计时。

防抖代码

let timer; // 维护同一个timer
function debounce(fn, delay) { 
   
    clearTimeout(timer);
    timer = setTimeout(function(){ 
   
        fn();
    }, delay);
}
// 用onmousemove测试一下防抖函数:
function testDebounce() { 
   
    console.log('test');
}
document.onmousemove = () => { 
   
    debounce(testDebounce, 1000);
}
----------------------------------------------
简化后代码:
function debounce(fn, delay) { 
   
    let timer; // 维护一个 timer
    return function () { 
   
        // let _this = this; // 取debounce执行作用域的this
        let args = arguments;
        if (timer) { 
   
            clearTimeout(timer);
        }
        // timer = setTimeout(function () { 
   
        // fn.apply(_this, args); // 用apply指向调用debounce的对象,相当于_this.fn(args);
        // }, delay);
        timer = setTimeout(()=> { 
   
            fn.apply(this, args); // 用apply指向调用debounce的对象,相当于this.fn(args);
        }, delay);
    };
}


// test
function testDebounce(e, content) { 
   
    console.log(e, content);
}
var testDebounceFn = debounce(testDebounce, 1000); // 防抖函数
document.onmousemove = function (e) { 
   
    testDebounceFn(e, 'debounce'); // 给防抖函数传参
}

什么是节流

即每隔一段时间,只执行一次函数。如游戏中的点击鼠标发射子弹,连续不断点按鼠标,并不会发射更多的子弹,而是按照一定的数量连续发射。

应用场景

1 滚动加载,加载更多或滚到底部监听
2 谷歌搜索框,搜索联想功能
3 高频点击提交,表单重复提交

函数防抖与节流的比较

都可以通过使用 setTimeout 实现。目的都是降低回调执行频率,节省计算资源。

节流代码

function throttle(fn, delay) { 
   
    let timer;
    return function () { 
   
        let _this = this;
        let args = arguments;
        if (timer) { 
   
            return;
        }
        timer = setTimeout(function () { 
   
            fn.apply(_this, args);
            timer = null; // 在delay后执行完fn之后清空timer,此时timer为假,throttle触发可以进入计时器
        }, delay)
    }
}


//test
function testThrottle(e, content) { 
   
    console.log(e, content);
}
let testThrottleFn = throttle(testThrottle, 1000); // 节流函数
document.onmousemove = function (e) { 
   
    testThrottleFn(e, 'throttle'); // 给节流函数传参
}
/** * 触发完事件 n 秒内不再触发事件,n秒后再执行 * 只执行最后一次点击 * @param event * @param time * @param flag 是否立即执行 * @returns {Function} * @constructor */
export function debounce(event, time, flag) { 
   
  const interval = time || 500
  let timer = null
  // 默认立即执行
  const f = flag || true
  return function(...args) { 
   
    clearTimeout(timer)
    if (f && !timer) { 
   
      event.apply(this, args)
    }
    timer = setTimeout(() => { 
   
      event.apply(this, args)
    }, interval)
  }
}

/** * 只在单位时间内执行一次 * 第一次事件不会触发,最后一次一定触发 * @param event * @param time * @returns {Function} * @constructor */
export function throttle(event, time) { 
   
  const interval = time || 500
  let timer = null
  return function(...args) { 
   
    if (!timer) { 
   
      timer = setTimeout(() => { 
   
        timer = null
        event.apply(this, args)
      }, interval)
    }
  }
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/210153.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 操作系统虚拟存储技术_虚拟存储

    操作系统虚拟存储技术_虚拟存储虚拟存储管理   在前面总结了集中存储管理的刚上,要求作业的逻辑地址空间连续的存放主存储器的某个区域中。当主存储器中没有足够大的区域是,则作业是无法装入的,或必须移动某些作业后才能装入。是否有可能吧作业的连续逻辑地址空间分散到几个不连续的主存区域,且仍能使作业正确执行呢?若可行的话,则可充分利用主存空间有可减少移动所花费的开销。不仅如此,还可采用虚拟存储管理技

    2022年9月25日
    3
  • 电信中兴f452光猫路由改桥接最简单的方式,亲自体验成功。

    电信中兴f452光猫路由改桥接最简单的方式,亲自体验成功。家里的光猫是中兴f452,默认是路由模式,改为桥接由路由器拨号,过程如下,验证成功。前提是去电信营业厅申请内网IP改公网IP,申请通过后再进行如下操作。1、网上的方法很多都是用超级用户密码登陆,也就是用户名telecomadmin密码nE7jA%5m,登录进去发现设置封装类型和连接模式的地方是灰色的,无法修改成桥接,此方法无效。2.尝试另一种方法,在可正常访问光猫192.168.1.1的情况下(不需登录),在浏览器直接访问http://192.168.1.1/bridge_route.gch,页

    2022年10月8日
    2
  • springboot 上传文件设置文件大小限制

    springboot 上传文件设置文件大小限制报错内容:org.springframework.web.multipart.MaxUploadSizeExceededException:Maximumuploadsizeexceeded;nestedexceptionisjava.lang.IllegalStateException:org.apache.tomcat.util.http.fileupload.impl.SizeLimitExceededException:therequestwasrejectedbe…

    2022年5月29日
    41
  • 使用MySQL实现分页查询[通俗易懂]

    使用MySQL实现分页查询[通俗易懂]本文关键字:MySQL、分页查询、真分页、假分页、LIMIT。在项目开发当中,经常要实现分页功能,在面试时也会经常被问到:什么是分页。这是因为在一个页面上能够显示的数据是有限的,而存放在数据库中的数据往往很多,我们必须将这些数据安放到不同的页面中去。

    2022年6月29日
    23
  • centos7配置sftp_怎么修改服务器sftp端口

    centos7配置sftp_怎么修改服务器sftp端口CentOS7修改sftp端口修改sftp默认端口为12322vi/etc/ssh/ssh_config把Port22改成Port12322vi/etc/ssh/sshd_config把Port22改成Port12322修改之后重启servicesshdrestart

    2025年11月17日
    1
  • 操作系统知识整理 – 进程控制块

    操作系统知识整理 – 进程控制块前提系统中需要有描述进程存在和能够反映其变化的物理实体,即进程的静态描述。进程的静态描述由3部分组成:进程控制块(ProcessControlBlock,PCB),有关程序段和该程序段操作的数据结构集。PCB是系统感知进程的唯一实体,用于描述进程的当前情况以及管理进程运行的全部信息,是操作系统中最重要的记录型数据结构。程序段以及数据结构集是进程完成所需功能的物质基础。一个进…

    2025年6月25日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号