找回密码
 会员注册
查看: 21|回复: 0

如何在Vue中使用防抖和节流

[复制链接]

2万

主题

0

回帖

6万

积分

超级版主

积分
64454
发表于 2024-10-12 00:00:06 | 显示全部楼层 |阅读模式
本文为 360 奇舞团前端工程师翻译原文:How to Debounce and Throttle Callbacks in Vue原文作者:Dmitri Pavlutin原文地址:https://dmitripavlutin.com/vue-debounce-throttle/在监听频繁触发的事件时,一定要多加小心,比如 用户在输入框打字、窗口大小调整、滚动、Intersection Observer 事件。这些事件总是被频繁触发,可能 几秒一次。如果针对每次事件都发起 fetch 请求(或类似的行为),那显然是不明智的。我们需要做的就是减缓事件处理程序的执行速度。这种缓冲技术就是 防抖(debounce) 和 节流(throttle) 。在本文中,你会了解到如何在 Vue 组件中 使用 防抖 和 节流 控制 观察者(watchers) 和 事件处理程序。1. 观察者 防抖我们先从一个简单的组件开始,我们的任务是 将用户输入到 文本框中的文本 输出到控制台: {{value}}在 输入框 敲几个字符。每次输入时,值就会被 log 到控制台。我们通过使用 观察者(watcher) 监听 value 数据属性 来实现了打印日志。但如果你想在 观察者的回调 中加入一个 使用 value 作为参数 的 GET 请求,那你应该不会期望太过频繁地发起请求。我们来对 打印控制台日志 这个行为做一下 防抖。核心思想是创建一个 防抖函数,然后在 观察者 内部调用该函数。我在这里选择了 'lodash.debounce' 的 防抖实现,但你可以自由选择喜欢的实现方式。我们来将 防抖逻辑 应用到组件: {{value}}如果你打开这个 demo,你会发现其实从用户角度来看,变化不大:你依旧可以像上一个 demo 中一样自由输入字符。但有一个区别:只有在最后一次输入的 500ms 之后,才会将新的输入值打印日志到控制台。这说明 防抖 在生效。观察者 的 防抖实现 只需要 3 个简单步骤:在 create() 钩子 里,创建 防抖回调,并将其赋值到实例上:this.debouncedWatch = debounce(..., 500)。在 观察者 回调 watch.value() { ... } 中 传入正确的参数 调用 this.debouncedWatch()。最后,beforeUnmount() 钩子中 调用 this.debouncedWatch.cancel() ,在卸载组件之前,取消所有还在 pending 的 防抖函数执行。采用同样的方式,你可以对任意数据属性的 观察者 应用 防抖。然后就可以安全执行 防抖回调内部的一些比较重的操作,比如 网络请求、繁重的 DOM 操作,等等。2. 事件处理器 防抖上面一节,我展示了如何对 观察者 使用 防抖,那么常规的事件处理器呢?我们重用之前用户输入数据到输入框的例子,但这一次会给输入框加个 事件处理器。像往常一样,如果你没有采取任何缓冲的措施,每当值被修改时,会被打印到控制台:打开这个 demo,在输入框打几个字符。看看控制台:你会发现每次你输入的时候就会有日志被打印出来。同样,如果你会执行一些比较重的操作(比如网络请求),可就不合适了。对 事件处理器 使用 防抖,可以参考下面这个:打开 demo,输入一些字符。组件只有在最后一次输入的 500ms 之后,才会将新的输入值打印日志到控制台。防抖 再一次生效了!事件处理器 的 防抖实现 只需要 3 个步骤:. 在 create() 钩子 里,创建实例后,立刻将 防抖回调 debounce(event => {...}, 500) 赋值到 this.debouncedHandler 。在输入框的 template 中 给 v-on:input 赋上 debouncedHandler :最后,在卸载组件之前, 在 beforeUnmount() 钩子中 调用 this.debouncedHandler.cancel() ,取消所有还在 pending 的 函数调用。另一方面,这些例子应用了 防抖 的技术。然而,同样的方式可以以用于创建 节流函数。3. 注意你可能不理解:为什么不直接在 组件的 method 选项中创建 防抖函数,然后在 template 中调用这些方法作为事件处理器?//...methods:{//WhynotdebouncedHandler:debounce(function(){...}},500)}//...这比在实例对象上创建 防抖函数 要简单的多。例如:这次不是在 created() 钩子 里创建 防抖回调了,而是将 防抖回调 赋给了 methods.debouncedHandler 。你如果试过 demo,你会发现是有效果的!问题是,组件使用 export default { ... } 导出的 options 对象,包括方法,会被组件实例重用。如果网页中有 2 个以上的组件实例,那么所有的组件都会应用 相同 的防抖函数 methods.debouncedHandler — 这会导致防抖出现故障。4. 总结在 Vue 中,可以很轻松的对 观察者 和 事件处理器 应用 防抖 和 节流。核心逻辑就是,在 created() 钩子 里,创建 防抖 或 节流 的回调,并赋值在实例上。//...created(){this.debouncedCallback=debounce((...args)=>{//Thedebouncedcallback},500);},//...A)然后在观察者内部调用实例上的防抖函数://...watch:{value(...args){this.debouncedCallback(...args);},},//...B)或在 template 中设定一个事件处理器:在这之后,每次调用 this.debouncedCallback(...args) ,就算执行频率非常高,内部的回调也能缓冲执行。你对 Vue 中的 防抖 和 节流 还什么问题吗?欢迎提问!-END-关于奇舞团奇舞团是 360 集团最大的大前端团队,代表集团参与 W3C 和 ECMA 会员(TC39)工作。奇舞团非常重视人才培养,有工程师、讲师、翻译官、业务接口人、团队 Leader 等多种发展方向供员工选择,并辅以提供相应的技术力、专业力、通用力、领导力等培训课程。奇舞团以开放和求贤的心态欢迎各种优秀人才关注和加入奇舞团。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 会员注册

本版积分规则

QQ|手机版|心飞设计-版权所有:微度网络信息技术服务中心 ( 鲁ICP备17032091号-12 )|网站地图

GMT+8, 2024-12-27 00:24 , Processed in 1.411166 second(s), 27 queries .

Powered by Discuz! X3.5

© 2001-2024 Discuz! Team.

快速回复 返回顶部 返回列表