React 事件监听踩坑:点一次按钮触发两次请求?原因竟然是这个…

在做项目的时候,经常会遇到一些「看似理所当然」的问题。最近我就在写一个模块时,被一个小问题坑了半天:点一下按钮,居然发了两次请求 🤯。

来,带大家复盘一下我是怎么踩坑、怎么排查,最后又是怎么解决的。


背景

场景很简单:页面里展示了一些markdown文本的信息,我需要把其中的电话号码识别出来,并在后面加个「📞 一键呼出」按钮,方便点击拨打。

于是我写了个 useEffect,在 document 上监听全局点击事件:

erlang 复制代码
useEffect(() => {
  console.log("注册一键呼出事件");
​
  const handleButtonClick = (event: any) => {
    if (event.target.classList.contains("phone-call")) {
      event.preventDefault();
      event.stopPropagation();
​
      const phoneNumber = event.target.getAttribute("data-phone");
      if (phoneNumber) {
        handlePhoneCall(phoneNumber);
      }
    }
  };
​
  document.addEventListener("click", handleButtonClick);
  return () => document.removeEventListener("click", handleButtonClick);
}, []);

按钮是这样插入到内容里的:

bash 复制代码
供应商电话:138****5678 
<button type="button" class="phone-call" data-phone="138****5678">📞一键呼出</button>

结果呢?点一次按钮,触发两次请求!


排查过程

1. 怀疑是事件冒泡

我的第一反应是:是不是点在按钮里的 emoji 也触发了一次?结果打印了一下 event.target,发现并没有,点按钮或者 emoji,都会只触发一次。

2. 怀疑是事件重复绑定

继续排查下去,发现和 React 18 的 StrictMode 有关。

👉 在开发模式下,StrictMode 会让 useEffect 的副作用执行两次: mount → unmount → mount

这意味着我的 document.addEventListener 实际上被绑定了两次。 于是点击按钮就触发了两次。

✅ 问题找到了。


解决方案

方案一:用 useCallback 固定函数引用

我第一步尝试是把 handleButtonClickuseCallback 包起来,避免清理不掉的问题。

ini 复制代码
const handleButtonClick = useCallback((event: MouseEvent) => {
  const target = (event.target as HTMLElement).closest(".phone-call");
  if (target) {
    event.preventDefault();
    event.stopPropagation();
    const phoneNumber = target.getAttribute("data-phone");
    if (phoneNumber) {
      handlePhoneCall(phoneNumber);
    }
  }
}, []);

这样能缓解一点,但 StrictMode 还是会执行两次,体验不算完美。


方案二:事件代理到组件容器(最终 ✅)

最终我选择了更彻底的方案 ------ 事件代理

不在 document 上绑全局事件,而是直接在组件最外层的容器(maven-scroll)上绑定 onClick,利用冒泡去捕获按钮点击:

ini 复制代码
<div
  className="maven-scroll"
  onClick={(e) => {
    const target = (e.target as HTMLElement).closest(".phone-call");
    if (target) {
      e.preventDefault();
      e.stopPropagation();
      const phoneNumber = target.getAttribute("data-phone");
      if (phoneNumber) {
        handlePhoneCall(phoneNumber);
      }
    }
  }}
>
  <MdContent mdContent={preprocessContent(supplierBaseReport?.companyInfo)} />
</div>

这样一来:

  • 点击按钮 → 被捕获,触发一次 ✅
  • 点击容器空白处 → closest 找不到,啥都不发生
  • 不用担心 StrictMode 下的 useEffect 双执行

完美解决 🎉。


最终效果

  1. 内容会被正则处理,手机号后面自动加按钮:

    bash 复制代码
    供应商电话:138****5678 
    <button type="button" class="phone-call" data-phone="138****5678">📞一键呼出</button>
  2. 点击按钮时,日志只输出一次


总结

  1. React 18 严格模式会导致副作用执行两次,如果你在副作用里做了事件绑定,要特别注意是否重复。
  2. 在类似场景下,事件代理比全局事件监听更优雅,逻辑也更可控。
  3. closest('.phone-call') 能确保即使点到按钮里的 emoji 或文字,也能正确识别。
  4. 别忘了给动态生成的 <button> 加上 type="button",避免表单里被当成提交按钮。
相关推荐
LaiYoung_1 分钟前
前端国际化适配提速 90%!这款 JS 脚本 CLI 工具,自动提中文、分模块、做替换,比 AI 更稳定
前端·javascript·人工智能
低代码布道师14 分钟前
CSS 伪类与伪元素:深度解析
前端·css
星月前端19 分钟前
css3元素倒影效果属性:box-reflect
前端·css·css3
vivo互联网技术27 分钟前
微信小程序端智能项目工程化实践
前端·人工智能·微信小程序·端智能
前端灵派派28 分钟前
openlayer轨迹回放
前端
Trust yourself24334 分钟前
如何获取easy-ui的表格的分页大小
前端·javascript·ui·oracle
zz-zjx41 分钟前
shell编程从0基础--进阶 1
linux·运维·前端·chrome·bash
濮水大叔1 小时前
能够动态推断与生成DTO是Node生态的一个重要里程碑
前端·typescript·node.js