[特殊字符] React 自定义 Hook 实现防抖(Debounce)

1.什么是防抖?

在前端开发中,我们经常会遇到一些高频触发的事件

  • 用户输入框输入 (onChange)

  • 页面滚动 (onScroll)

  • 窗口大小变化 (resize)

这些事件如果每次触发都立即执行回调,会带来性能问题。

防抖(Debounce) 的思想就是: 👉 在一段时间内,多次触发只执行最后一次。

打个比方:

  • 你在搜索框里疯狂输入,但只想在用户 停下输入 1 秒 后再真正发起搜索请求。

  • 这样就避免了每打一个字都请求一次后端。


2常见问题

如果我们不做防抖,代码可能像这样:

javascript 复制代码
<input onChange={(e) => fetchData(e.target.value)} />

每输入一个字符,就会触发一次 fetchData,非常浪费资源。 所以我们需要封装一个防抖逻辑

3.用 React 自定义 Hook 实现防抖

javascript 复制代码
import { useRef,useEffect } from "react";
export function useFangdou(fn,delay){
    const time = useRef(null)
    const callback=(a)=>{
        if(time.current) clearTimeout(time.current)
        time.current=setTimeout(()=>{
            fn(a)
        },delay*1000)
    }
    return callback
}

清除上一个定时器,然后执行我们的希望变成防抖的函数执行。

关键点解释

  1. useRef 保存定时器

    1. React 每次渲染函数都会重新执行,但 useRef 保存的值不会丢失。

    2. 所以我们可以用它来存储 setTimeout 的返回值。

  2. 清理上一次定时器

    1. 如果在延迟时间内再次触发,先 clearTimeout,再开新的 setTimeout

    2. 这样只会执行最后一次。

  3. 返回一个函数

    1. 外部直接拿到 useFangdou 返回的 callback 去用即可。

4.使用实例

javascript 复制代码
import React, { useEffect, useState, useRef } from "react";
import { useFangdou } from "../../stils/useFangdou";
import "./list.css";
export default function List() {
  const go = (e) => {
    console.log(e.target.value);
  };
  const scroll = (e) => {
    console.log(" ", e);
  };
  const fangdou = useFangdou(scroll, 1);
  const fangdou2 = useFangdou(go, 2);
  return (
    <div>
      <div
        className="box"
        onScroll={(e) => {
          fangdou(e);
        }}
      >
        <div className="child"></div>
      </div>
      <input
        type="text"
        onChange={(e) => {
          fangdou2(e);
        }}
      />
    </div>
  );
}

这样逻辑就是:

  • 每次 text 变化,先清掉上一次的定时器,再开启一个新的 setTimeout

  • 2 秒后才执行 console.log

  • 相当于实现了一个输入框的 防抖 (debounce)

相关推荐
MediaTea26 分钟前
Python 第三方库:lxml(高性能 XML/HTML 解析与处理)
xml·开发语言·前端·python·html
西陵30 分钟前
Nx带来极致的前端开发体验——使用MF进行增量构建
前端·javascript·架构
Nicholas6842 分钟前
flutter滚动视图之ProxyWidget、ProxyElement、NotifiableElementMixin源码解析(九)
前端
JackieDYH1 小时前
vue3中reactive和ref如何使用和区别
前端·javascript·vue.js
ZZHow10242 小时前
React前端开发_Day4
前端·笔记·react.js·前端框架·web
前端开发爱好者2 小时前
弃用 html2canvas!快 93 倍的截图神器
前端·javascript·vue.js
ss2732 小时前
手写MyBatis第39弹:深入MyBatis BatchExecutor实现原理与最佳实践
前端·javascript·html
leon_teacher3 小时前
HarmonyOS权限管理应用
android·服务器·前端·javascript·华为·harmonyos
lumi.3 小时前
HarmonyOS image组件深度解析:多场景应用与性能优化指南(2.4详细解析,完整见uniapp官网)
前端·javascript·小程序·uni-app·html·css3
OEC小胖胖3 小时前
动态UI的秘诀:React中的条件渲染
前端·react.js·ui·前端框架·web