在JavaScript中的防抖函数 - 通过在React中构建自动完成功能来解释

当你将一个新应用推向生产环境时,你希望确保它用户友好。网站的性能是用户体验的关键部分。每个用户都希望网站及其内容能够快速加载。每一秒都是宝贵的,可能导致用户再也不会访问你的网站。

在本指南中,我们将了解JavaScript中一个非常重要的技术,即防抖函数。然后,我将向您展示如何在React中使用防抖实现自动完成功能。

(本文内容参考:java567.com)

目录:

  • 什么是防抖函数?
  • 如何在JavaScript中实现防抖?
  • 防抖函数的用例
  • 结论

什么是防抖函数?

防抖是一种通过控制函数执行的时间来改善功能性能的策略。

防抖接受一个函数并将其转换为更新后(防抖的)函数,以便在一定时间后执行原始函数中的代码。

如果在该时间段内再次调用防抖函数,则会重置先前的计时器,并为此函数调用启动一个新的计时器。这个过程对每个函数调用都会重复。

一个例子会让你更好地理解。让我们拿一个函数fun()来说。我们希望在500毫秒后执行这个函数。

function fun() {console.log('This is a function')
}

经过防抖处理后,会返回一个新的函数debouncedFun()。现在,每当你调用debouncedFun()时,它都会在500毫秒后执行。

如果你在第一次调用后的500毫秒内再次调用它,先前的计时器会被重置,并为第二次函数调用启动一个新的计时器。如果你在500毫秒内保持调用函数,这个过程会重复。

如何在JavaScript中实现防抖?

让我们了解如何在JavaScript中实现防抖。首先,我们将讨论我们的需求。我们希望防抖函数的行为是什么?

  • 延迟一定时间执行函数。
  • 如果再次调用函数,则重置计时器。

为了防抖一个函数,我们将有一个单独的函数,它接受函数引用和延迟作为参数,并返回一个防抖函数。

function debounce(func, delay) {return () => {}   // 返回防抖函数
}

这个函数只会被调用一次,以返回一个防抖函数,并且这个防抖函数将在后续的代码中使用。

要延迟一段时间执行函数,我们可以简单地在JavaScript中使用setTimeout函数。

function debounce(func, delay) {return () => {setTimeout(() => {func()}, delay)}
}

这会将函数调用延迟delay毫秒。但是,这还不完整,因为它只满足了第一个要求。我们如何实现第二个行为呢?

让我们创建一个变量timeout,并将其赋值为setTimeout方法的返回值。setTimeout方法返回一个唯一的标识符给timeout,这个标识符由timeout变量持有。

function debounce(func, delay) {let timeout=nullreturn () => {timeout=setTimeout(() => {func()}, delay)}
}

每次你调用setTimeout时,ID都是不同的。我们将使用这个timeout变量来重置计时器。

但是我们如何从debounce()方法外部访问timeout呢?如前所述,debounce()方法只被调用一次来返回一个防抖函数。这个防抖函数再执行防抖逻辑。

那么,即使在debounce()函数之外使用debounced函数,它是如何访问timeout的呢?好吧,它使用了一个叫做闭包的概念。

JavaScript中的闭包是什么?

在JavaScript中,内部函数始终可以访问外部函数的局部变量。在我们的例子中,内部函数可以访问在debounce()方法中具有函数级作用域的timeout变量。

但是当外部函数返回这个内部函数时,即使外部函数已经执行完毕,内部函数仍然持有对外部函数局部变量的引用。这就是闭包的概念。

让我们用一个例子来理解闭包。

function outerFunction() {const x = 5;return () => {console.log(x);}
}const inner = outerFunction();inner(); // 输出5// console.log(x)   抛出引用错误

在这里,如果我们调用inner(),代码将没有错误并且打印5。但是,如果我们试图直接访问x,JavaScript将抛出一个引用错误。

在这里插入图片描述
在JavaScript中引用错误

在这里,inner()封闭了x,只有这个函数可以使用这个变量,其他的函数无法访问它。

回到防抖函数

让我们回到我们离开的地方:

function debounce(func, delay) {let timeout=nullreturn () => {timeout=setTimeout(() => {func()}, delay)}
}

在这里,JavaScript使用闭包来在每次使用防抖函数时都保持对timeout的访问。

让我们利用这一点。由于debouncedFun()在每次函数调用中都可以访问相同的timeout变量,我们可以添加一个条件来检查先前的计时器是否存在。我们可以使用一个空值检查来实现这一点,if(timeout !== null)或if(timeout)。

然后,我们使用clearTimeout()方法来取消上一个计时器,从而重置计时器。

在启动新的计时器之前,添加以下语句:

if(timeout) clearTimeout(timeout)

计时器被重置后,为当前函数调用启动一个新的计时器,并将其ID分配给timeout。对于由于闭包而访问相同timeout的后续函数调用,这个过程会重复。

function debounce(func, delay) {let timeout=nullreturn () => {if(timeout) clearTimeout(timeout)timeout=setTimeout(() => {func()}, delay)}
}

通过这样做,我们满足了我们的第二个要求 - 即重置计时器并启动新的计时器。现在是时候使用这个防抖函数了。

让我们将fun()传递给debounce()方法,延迟500ms。

const debouncedFun = debounce(fun, 500)

debouncedFun()基本上是带有防抖行为的fun()。让我们以不同的时间间隔调用这个函数来测试我们的功能。

debouncedFun()setTimeout(debouncedFun, 300)setTimeout(debouncedFun, 900)

第一个函数调用立即执行。另外两个分别在300ms和900ms后执行。你能猜到输出吗?

代码打印出两次"This is a function"。让我们来理解为什么。在这里,第一次调用后,fun()被安排在500ms后执行。但第二次调用在300ms内,这会重置计时器并启动一个新的计时器。

500ms已经过去,fun()方法执行。然后,在900ms时,又发生了另一个函数调用。这会再次在500ms后执行fun()。

我们还应该做一点小小的改进。我们的逻辑没有考虑函数参数。让我们用带有参数的fun()来替换它。

function fun(a, b) {console.log(`This is a function with arguments ${a} and ${b}`)
}

为了在防抖时包含参数,返回一个接受参数的防抖函数。

function debounce(func, delay) {let timeout=nullreturn (...args) => {if(timeout) clearTimeout(timeout)timeout=setTimeout(() => {func(...args)timeout=null}, delay)}
}

通过使用展开运算符,传递给防抖函数的任何参数都将存储在args变量中的数组中。然后,使用相同的args数组展开调用实际函数。

const debouncedFun=debounce(fun, 500)
debouncedFun(2,3)

上面的代码在500毫秒后打印出"This is a function with arguments 2 and 3"。

防抖函数的用例

让我们看看防抖在实际应用中是如何使用的。防抖最常见的用例是自动完成功能。你一定见过很多网站,在输入框中输入内容时,它会显示一个结果列表。

这里有一个来自Google搜索的例子:

在这里插入图片描述

Google搜索在输入"Top 10"后的自动完成功能

Google搜索显示最近和常见的搜索项。这些信息主要来自浏览器缓存。但是,一些网站会向后端服务器发出API调用,从数据库中获取数据。

通过在输入元素上添加一个onchange事件并在事件处理程序中实现fetch逻辑,可以很容易地实现这一点。但这里有一个小问题。

考虑以下例子:

在这里插入图片描述

每次输入值都会进行API请求

当我输入单词absolute时,每次输入字段的值更改都会进行一次API请求。在很短的时间内我们进行了8次API请求,这会给后端服务器造成很大的负载,并可能导致性能问题。

理想情况下,我们希望在用户完成输入后一段时间内显示自动完成功能的结果。在这里,用户一次性输入了absolute,所以我们不需要在每次输入更改时都显示结果,而是在用户完成输入后显示结果 - 也就是说,在输入更改和结果显示之间添加一些延迟。

所以,我们只有在用户完成输入时才进行API调用,而不是在每次输入更改时都调用。这减少了API调用的数量并提高了性能。我们可以使用防抖来实现这种行为。

让我们了解如何在React中实现自动完成功能。

自动完成功能示例

使用create-react-app(或像Vite这样的现代构建工具)创建项目。删除现有的样板代码。不需要安装额外的依赖项。运行npm start命令启动项目。您可以在GitHub上找到完整的代码。

我已经设置了一个Node服务器来为应用程序获取数据。您可以在Git存储库中找到它。运行node server命令来启动它。我不打算展示Node.js代码,因为它超出了本教程的范围。

让我们开始实现。我们将编写一个简单的自动完成功能。该应用程序应该显示一个包含用户输入字符串的城市

列表。

应用程序组件

我们首先需要一个输入元素来接受用户输入,以及一个用于搜索结果的结果容器。将一个异步函数作为事件处理程序附加到输入元素上。

function App() {const [data, setData] = useState(null)const loadData = async (event) => {}return (<div className="App"><input type="text" onChange={(e) => loadData(e)}/>{data && data.length !== 0 &&<div className="results-container">{data.map(item => (<div key={item.id} className="result-item"><p> {item.city} </p></div>))}</div>}</div>);
}

数据将存储为状态,并且只有在数据非空时才会显示结果。我会跳过本教程的CSS,你可以在Git Repo中找到它。

事件处理程序

loadData()函数获取我们的数据并将响应存储为状态。

const loadData = async (event) => {const value=event.target.valueif(value === '') {setData(null)return}const response=await fetch(`http://localhost:8000/data/${value}`)const res=await response.json()setData(res)
}

如果没有输入值,简单地退出函数。否则,对节点服务器端点进行请求。由于此函数在每次输入更改时都会被调用,因此我们将对该函数进行防抖。

使用自定义钩子实现防抖

我们将在一个自定义钩子中编写防抖逻辑。自定义钩子的优点是您可以在整个应用程序中重用相同的逻辑。强烈建议您这样做。

创建一个新文件夹custom-hooks,在其中创建一个名为useDebounce.js的文件。如前所述,useDebounce()方法应该接受一个函数和延迟作为参数,并返回防抖函数。

const useDebounce = (func, delay) => {let timeout=nullreturn (...args) => {if(timeout) clearTimeout(timeout)timeout=setTimeout(() => {func(...args)}, delay)}
}export default useDebounce

现在,在应用程序组件内部,调用此方法一次以获取loadDataDebounced()

const loadDataDebounced = useDebounce(loadData, 400)

我们将使用这个新方法作为输入元素的事件处理程序。

<input type="text" onChange={(e) => loadDataDebounced(e)}/>
输出

在输入元素内输入搜索字符串以测试我们的代码。

在这里插入图片描述

屏幕上的输出

在这里插入图片描述

正如你在网络选项卡中看到的,只发送了一个请求,而不是三个。这使得搜索性能得到了很大的改善。

结论

在本教程中,您了解了什么是防抖,以及如何实现它。防抖通过延迟一定时间来执行函数,并在函数再次调用时重置前一个计时器来延迟函数执行。

防抖使用了一个重要的概念 - 闭包。我稍微偏离了实现来解释闭包是什么。这对于初学者来说可能是一个令人困惑的概念,所以请花些时间来理解它。闭包允许您在函数执行完毕后继续使用局部变量。

之后,我向您展示了防抖的一个常见用例 - 自动完成功能。通过防抖可以提高功能的性能。我还向您展示了如何在React中实现自动完成功能,并使用自定义钩子来实现防抖。希望这对您未来的项目有所帮助。

(本文内容参考:java567.com)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/686640.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2024.2.15 模拟实现 RabbitMQ —— 消息持久化

目录 引言 约定存储方式 消息序列化 重点理解 针对 MessageFileManager 单元测试 小结 统一硬盘操作​​​​​​​ 引言 问题&#xff1a; 关于 Message&#xff08;消息&#xff09;为啥在硬盘上存储&#xff1f; 回答&#xff1a; 消息操作并不涉及到复杂的增删查改消…

人工智能学习与实训笔记(十四):Langchain之Agent

人工智能专栏文章汇总&#xff1a;人工智能学习专栏文章汇总-CSDN博客 本篇目录 0、概要 1、Agent整体架构 2、langchain中agent实现 3、Agent业务实现逻辑 0、概要 Agent是干什么的&#xff1f; Agent的核心思想是使用语言模型&#xff08;LLM&#xff09;作为推理的大脑…

redis为什么使用跳跃表而不是树

Redis中支持五种数据类型中有序集合Sorted Set的底层数据结构使用的跳跃表&#xff0c;为何不使用其他的如平衡二叉树、b树等数据结构呢&#xff1f; 1&#xff0c;redis的设计目标、性能需求&#xff1a; redis是高性能的非关系型&#xff08;NoSQL&#xff09;内存键值数据…

【51单片机实验笔记】开关篇(二) 矩阵按键

目录 前言原理图分析矩阵按键扫描算法 软件实现1. 矩阵键盘检测2. 简易计算器实现 总结 前言 本节内容&#xff0c;我们学习一下矩阵按键&#xff0c;它是独立按键的阵列形式&#xff0c;常见的应用即键盘。 本节涉及到的封装源文件可在《模块功能封装汇总》中找到。 本节完…

websocket数据帧格式

客户端、服务端数据的交换&#xff0c;离不开数据帧格式的定义。因此&#xff0c;在实际讲解数据交换之前&#xff0c;我们先来看下WebSocket的数据帧格式。 WebSocket客户端、服务端通信的最小单位是帧&#xff08;frame&#xff09;&#xff0c;由1个或多个帧组成一条完整的消…

基于协同过滤的时尚穿搭推荐系统

项目&#xff1a;基于协同过滤的时尚穿搭推荐系统 摘 要 基于协同过滤的时尚穿搭推荐系统是一种能自动从网络上收集信息的工具&#xff0c;可根据用户的需求定向采集特定数据信息的工具&#xff0c;本项目通过研究服饰流行的分析和预测的分析和预测信息可视化时尚穿搭推荐系统…

C++中的volatile:穿越编译器的屏障

C中的volatile&#xff1a;穿越编译器的屏障 在C编程中&#xff0c;我们经常会遇到需要与硬件交互或多线程环境下访问共享数据的情况。为了确保程序的正确性和可预测性&#xff0c;C提供了关键字volatile来修饰变量。本文将深入解析C中的volatile关键字&#xff0c;介绍其作用、…

浅谈电商场景中的扣除库存问题

库存 一、场景二、扣减时机1.下单时扣库存2.支付完成扣库存3.预扣除 三、库存存储方案1.数据库存储2.数据库缓存混合存储 四、整体方案1.单数据库方案2.主从数据库方案3.主从数据库缓存方案4.数据库缓存混合存储 五、其他情况1.秒杀QPS过高2.Redis QPS过高3.Master DB QPS过高4…

使用ShardingJDBC实现分库分表

一、测试环境 JDK&#xff1a;1.8SpringBoot&#xff1a;2.7.17MySQL驱动&#xff1a;5.1.49MyBatis&#xff1a;2.3.1shardingJDBC&#xff1a;5.1.0 二、核心依赖 <!-- mysql 驱动 --> <dependency><groupId>mysql</groupId><artifactId>mysq…

Manifest merger failed with multiple errors, see logs

问题 Manifest merger failed with multiple errors, see logs详细问题 笔者进行Android 项目开发&#xff0c;修改AndroidManifest.xml代码后&#xff0c;控制台报错 AndroidManifest.xml报错核心代码 <manifest><uses-permission android:name"android.perm…

【C语言】长篇详解,字符系列篇1-----“混杂”的各种字符类型字符转换和strlen的模拟实现【图文详解】

欢迎来CILMY23的博客喔&#xff0c;本期系列为【C语言】长篇详解&#xff0c;字符系列篇1-----“混杂”的各种字符函数……&#xff0c;图文讲解各种字符函数&#xff0c;带大家更深刻理解C语言中各种字符函数的应用&#xff0c;感谢观看&#xff0c;支持的可以给个赞哇。 前言…

内存块与内存池

&#xff08;1&#xff09;在运行过程中&#xff0c;MemoryPool内存池可能会有多个用来满足内存申请请求的内存块&#xff0c;这些内存块是从进程堆中开辟的一个较大的连续内存区域&#xff0c;它由一个MemoryBlock结构体和多个可供分配的内存单元组成&#xff0c;所有内存块组…

Java学习笔记------static

static 创建Javabean类 public class student {private int age;private String name;private String gender;public student() {}public student(int age, String name, String gender) {this.age age;this.name name;this.gender gender;}/*** 获取* return age*/public…

使用Python编写脚本-根据端口号杀掉进程

我的GitHub&#xff1a;Powerveil - GitHub 我的Gitee&#xff1a;Powercs12 - Gitee 皮卡丘每天学Java 从前段开始遇到一个问题&#xff0c;服务在启动的时候总是端口被占用&#xff0c;发现还是Java程序&#xff0c;但是当时并没有启动Java程序&#xff0c;电脑出问题了。 一…

【Linux】Framebuffer 应用

# 前置知识 LCD 操作原理 在 Linux 系统中通过 Framebuffer 驱动程序来控制 LCD。 Frame 是帧的意思&#xff0c; buffer 是缓冲的意思&#xff0c;这意味着 Framebuffer 就是一块内存&#xff0c;里面保存着一帧图像。 Framebuffer 中保存着一帧图像的每一个像素颜色值&…

Tomcat要点总结

一、Tomcat 服务中部署 WEB 应用 1.什么是Web应用 &#xff08;1&#xff09; WEB 应用是多个 web 资源的集合。简单的说&#xff0c;可以把 web 应用理解为硬盘上的一个目录&#xff0c; 这个目录用于管理多个 web 资源。 &#xff08;2&#xff09;Web 应用通常也称之为…

七、ActiveMQ的传输协议

ActiveMQ的传输协议 一、是什么二、协议1.TCP(默认)2.NIO3.AMQP4.STOMP5.SSL6.MQTT7 WS 三、NIO配置案例1.修改activemq.xml2.重启3.生产者/消费者4.性能提升4.1 配置4.2 生产者/消费者 一、是什么 官网地址&#xff1a;http://activemq.apache.org/configuring-version-5-tra…

Mysql知识点汇总

Mysql知识点汇总 1. Mysql基本场景的简单语句。2. Mysql的增删改查&#xff0c;统计表中的成绩最好的两个同学的名字&#xff0c;年级等。3&#xff1a;请使用多种方法查询每个学生的每门课分数>80的学生姓名4、order by&#xff0c;group by&#xff0c;子查询4.1、having和…

Apache Httpd 常见漏洞解析(全)

一、Apache HTTPD 换行解析漏洞 漏洞编号&#xff1a;CVE-2017-15715 Apache HTTPD是一款HTTP服务器&#xff0c;它可以通过mod_php来运行PHP网页。 其2.4.0~2.4.29版本中存在一个解析漏洞。 在解析PHP时&#xff0c;1.php\x0A将被按照PHP后缀进行解析&#xff0c;导致绕过…

XSS数据接收平台

一.使用xss数据接收平台的好处&#xff1a; 正常执行反射型xss和存储型xss&#xff0c;反射型xss在执行poc时&#xff0c;会直接在页面弹出执行注入的poc代码&#xff1b;存储型则是&#xff0c;在将poc代码注入用户的系统中后&#xff0c;用户访问有存储型xss的地方&#xff…