新闻中心

解决React中useEffect重复执行的问题

2025-10-13
浏览次数:
返回列表

解决react中useeffect重复执行的问题

React开发者经常遇到useEffect钩子意外执行两次的情况,尤其是在开发模式下。本文将深入探讨useEffect重复执行的原因,并提供有效的解决方案,确保你的副作用函数按预期运行,同时优化加载状态的管理,避免不必要的数据库操作。

为什么useEffect会执行两次?

在React 18及更高版本中,严格模式(Strict Mode)默认启用,尤其是在开发环境下。严格模式旨在帮助开发者尽早发现潜在问题,它会故意触发某些副作用函数两次,以暴露不纯的渲染和潜在的错误。这通常是useEffect重复执行的罪魁祸首。

如何解决useEffect重复执行问题?

虽然严格模式是导致useEffect重复执行的主要原因,但我们通常不建议直接禁用它,因为它有助于发现潜在的bug。相反,我们应该修改我们的代码,使其能够正确处理重复执行的情况。以下是一些有效的策略:

1. 幂等性操作

确保你的副作用函数是幂等的。这意味着无论函数执行多少次,其结果都应该相同。例如,如果你的useEffect用于向数据库发送请求,确保你的API能够处理重复请求,或者在客户端实现去重逻辑。

2. 利用useRef存储状态

如果useEffect中的副作用是基于某个状态的,你可以使用useRef来存储该状态,并在每次执行副作用之前检查该状态是否已经更新。useRef创建的是一个可以在组件的整个生命周期中保持不变的变量,且修改useRef不会触发组件重新渲染。

3. 移除或调整严格模式

如果确认严格模式是导致问题的唯一原因,并且无法修改副作用函数以适应重复执行,可以考虑暂时移除或调整严格模式的范围。但请记住,这可能会隐藏潜在的问题,因此应谨慎使用。

4. 优化依赖项

检查useEffect的依赖项数组。确保只包含真正需要监听的状态或props。不必要的依赖项会导致useEffect在不必要的时候执行。

PHP轻论坛 PHP轻论坛

简介PHP轻论坛是一个简单易用的PHP论坛程序,适合小型社区和个人网站使用。v3.0版本是完全重构的版本,解决了之前版本中的所有已知问题,特别是MySQL保留字冲突问题。主要特点• 简单易用:简洁的界面,易于安装和使用• 响应式设计:适配各种设备,包括手机和平板• 安全可靠:避免使用MySQL保留字,防止SQL注入• 功能完善:支持分类、主题、回复、用户管理等基本功能• 易于扩展:模块化设计,便于

PHP轻论坛 26 查看详情 PHP轻论坛

示例代码分析与改进

以下面的代码为例,展示如何解决useEffect重复执行以及加载状态管理的问题:

import { type AppType } from 'next/app';
import { api } from '~/utils/api';
import '~/styles/globals.css';
import N* from '~/components/N*';
import { useEffect, useState } from 'react';

const MyApp: AppType = ({ Component, pageProps }) => {
  const [showCart, setShowCart] = useState(false);
  const [loading, setLoading] = useState(false);

  const createSession = api.user.createSession.useMutation();

  const generateId = async (): Promise<string | undefined> => {
    const res = await createSession.mutateAsync();
    if (res.response) {
      return res.response.id;
    } else if (res.error) {
      return res.error;
    }
  };

  const setSessionId = async () => {
    const tmp: string | undefined = await generateId();
    if (tmp) document.cookie = `sessionId=${tmp}`;
    setLoading(false);
  };

  useEffect(() => {
    if (!loading) {
      setLoading(true);
      const cookieString: string = document.cookie;

      const cookies: string[] = cookieString.split(';') || [];

      let sessionId: string | null = null;

      for (let i = 0; i < cookies.length; i++) {
        const cookie: string | undefined = cookies[i];

        if (!cookie || cookie.trim() === '') {
          continue;
        }

        if (cookie.trim().startsWith('sessionId=')) {
          sessionId = cookie.trim().substring('sessionId='.length);
          break;
        }
      }
      if (!sessionId) {
        void setSessionId();
      }
    }
  }, []);

  return (
    <>
      <N* showCart={showCart} setShowCart={setShowCart} />
      <h1>{loading && 'LOADING'}</h1>
      <Component {...pageProps} />
    </>
  );
};

export default api.withTRPC(MyApp);

问题分析:

  1. useEffect依赖项为空,导致组件每次渲染都会执行。
  2. loading状态的设置和使用不一致,导致加载状态显示不正确。

改进后的代码:

import { type AppType } from 'next/app';
import { api } from '~/utils/api';
import '~/styles/globals.css';
import N* from '~/components/N*';
import { useEffect, useState } from 'react';

const MyApp: AppType = ({ Component, pageProps }) => {
  const [showCart, setShowCart] = useState(false);
  const [loading, setLoading] = useState(true); // 初始化loading为true

  const createSession = api.user.createSession.useMutation();

  const generateId = async (): Promise<string | undefined> => {
    const res = await createSession.mutateAsync();
    if (res.response) {
      return res.response.id;
    } else if (res.error) {
      return res.error;
    }
  };

  const setSessionId = async () => {
    const tmp: string | undefined = await generateId();
    if (tmp) document.cookie = `sessionId=${tmp}`;
    setLoading(false);
  };

  useEffect(() => {
    const getSessionId = () => {
      const cookieString: string = document.cookie;
      const cookies: string[] = cookieString.split(';') || [];

      let sessionId: string | null = null;

      for (let i = 0; i < cookies.length; i++) {
        const cookie: string | undefined = cookies[i];

        if (!cookie || cookie.trim() === '') {
          continue;
        }

        if (cookie.trim().startsWith('sessionId=')) {
          sessionId = cookie.trim().substring('sessionId='.length);
          break;
        }
      }

      return sessionId;
    };

    if (!getSessionId()) {
      void setSessionId();
    } else {
      setLoading(false); // 如果sessionId已存在,则设置loading为false
    }
  }, []);

  return (
    <>
      <N* showCart={showCart} setShowCart={setShowCart} />
      {loading && <h1>LOADING</h1>} {/* 仅当loading为true时显示 */}
      <Component {...pageProps} />
    </>
  );
};

export default api.withTRPC(MyApp);

改进说明:

  1. 初始化 loading 状态: 将 loading 的初始状态设置为 true,确保在组件挂载时显示加载状态。
  2. 移除不必要的 loading 判断: 在 useEffect 内部移除了 if (!loading) 的判断,因为初始时 loading 已经是 true,并且在获取或创建 sessionId 后会设置为 false。
  3. useEffect 依赖项: 保持依赖项为空数组 [],确保 useEffect 只在组件挂载时执行一次。

总结

useEffect重复执行是React开发中常见的问题,但通过理解其原因并采取适当的措施,可以有效地解决这个问题。关键在于理解严格模式的作用,确保副作用函数的幂等性,并优化依赖项。同时,正确管理加载状态,可以提升用户体验。通过本文提供的策略和示例代码,相信你能更好地掌握useEffect的使用,避免重复执行带来的问题。

以上就是解决React中useEffect重复执行的问题的详细内容,更多请关注其它相关文章!


# react  # cookie  # app  # ppt  # css  # 如何解决  # seo其他营销方式  # 为空  # 黑帽seo工  # 杂志网站建设工作汇报  # 进一步优化网站栏目设置  # 贵州网站建站建设推荐  # 德国站seo  # 如何做好品牌营销推广  # 哪里的seo优化较好  # 永康网站建设的主要特点  # 启策seo  # 易用  # 设置为  # 自定义  # 是在  # 复选框  # 移除  # 两次  # 加载  # 为什么  # 开发环境  # ai  # session 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: 俄罗斯方块最新版入口 俄罗斯方块在线玩官网入口  “在文档元素之后找到了标记”是什么错误? 检查并修复XML中多个根元素的3个方法  C++的std::forward_list怎么用_C++ STL中单向链表容器的特点与应用  sublime如何配置Go语言开发环境_sublime搭建Golang编译运行系统  Golang如何通过reflect获取匿名字段方法_Golang reflect匿名字段方法访问技巧  微信网页版登录教程_微信网页版登录入口在哪  魅族20怎样在浏览器开无图省流_iPhone魅族20浏览器开无图省流【流量节省】  必由学官方登录入口 必由学教师学生账号快速访问  Tabulator表格日期时间排序问题及自定义解决方案  sublime如何配置Python开发环境_将sublime打造成轻量级Python IDE  如何在Promise链中优雅地中断后续then执行  2026春节假期时间安排 2026春节假日查询  C++如何连接MySQL数据库_C++使用Connector/C++操作MySQL数据库教程  QQ邮箱官网登录入口 QQ邮箱网页版邮箱快速登录  Spring Boot内嵌服务器与J*a EE全栈特性:选择与部署策略  移动端XML文件怎么转换成Excel 手机和平板上的解决方案  AO3官方镜像站点汇总 AO3同人作品网页版直达链接  R星幕后开发视频泄露 包含《GTA6》等多款大作  深入理解Promise链:如何在catch后中断then的执行  163邮箱注册官网 免费申请163个人邮箱  C++如何使用AddressSanitizer(ASan)_C++调试工具中检测内存访问错误的利器  UC浏览器网页版登录入口官网 电脑版网址入口  win11 arm版怎么安装 M1/M2 Mac虚拟机安装ARM win11的方法  中兴BladeV30怎样用测距估书架层高_iPhone中兴BladeV30测距估书架层高【家装参考】  海量存储:机器视觉智能化的核心基石  PHP 枚举:根据字符串获取枚举案例的策略与实现  J*aScript中赋值与自增运算符的复杂交互与执行机制  Google翻译怎么语音输入_Google翻译语音输入功能使用与设置方法  在J*a中如何开发在线活动报名与管理系统_活动报名管理项目实战解析  Win11 BitLocker密码忘了怎么办 Win11找回BitLocker恢复密钥方法【解决】  将HTML Canvas内容转换为可上传的图像文件(File对象)  铁路12306的积分有效期是多久_铁路12306积分有效期说明  Mac怎么查看崩溃日志_Mac控制台错误报告分析  mc.js官网登录入口 mc.js官方登录入口最新版  没有大陆身份证/银行卡如何实名微信? 亲测有效的几种方法分享  html网页设计源代码怎么运行_运行html网页设计源代码步骤【指南】  C++编译期如何执行复杂计算_C++模板元编程(TMP)技巧与应用  CSS子选择器:如何区分并样式化嵌套列表的子层级  蛙漫限时开放最深处链接_蛙漫全站漫画会员同款秒开地址  126邮箱账号注册 电脑版登录入口  如何提高微信支付的安全性_微信支付安全防护与设置建议  JUnit5/Mockito:优雅测试内部依赖与异常处理的实践  React中useState与局部变量:理解组件状态管理与渲染机制  qq邮箱发邮件给国外发不出去_QQ邮箱国际邮件发送失败原因与解决  Centos/Linux 系统下安装 composer 的完整步骤  AO3官方可用镜像 Archive of Our Own网页版最新入口  包子漫画官方网站在线链接-包子漫画在线阅读平台主页地址  qq浏览器如何查看和导出已保存的密码 qq浏览器密码管理器数据备份教程  铁路12306官网网页端快速入口 铁路12306官方首页登录教程  QQ邮箱正确登录入口_QQ邮箱官方网站使用地址 

搜索