0

0

避免JavaScript Fetch请求重复发送的常见陷阱

聖光之護

聖光之護

发布时间:2025-12-03 12:08:38

|

715人浏览过

|

来源于php中文网

原创

避免JavaScript Fetch请求重复发送的常见陷阱

本文旨在探讨javascript中fetch api请求意外重复发送的常见原因及解决方案。通过分析将异步请求逻辑错误地放置在循环内部的场景,并结合实际代码示例,详细阐述如何重构代码以确保fetch请求按预期执行,从而避免服务器端重复处理和客户端潜在的网络错误。

在现代Web开发中,JavaScript的Fetch API是进行网络请求的强大工具。然而,开发者有时会遇到Fetch请求意外地被重复触发的问题,这不仅可能导致服务器端数据处理异常,还可能在客户端引发诸如“NetworkError”之类的错误。本文将深入分析这类问题的一个常见根源,并提供专业的解决方案。

Fetch请求重复触发的典型场景

当一个按钮被点击时,我们通常期望与之关联的JavaScript函数只执行一次,并且其中的Fetch请求也只发送一次。然而,在某些情况下,即使按钮只点击了一次,Fetch请求却可能在后台被多次发送。

考虑以下一个使用Django后端和JavaScript前端的场景:

前端JavaScript代码:

立即学习Java免费学习笔记(深入)”;

function post_db() {
  // 其他初始化代码
  async function postData() {
    const url = "fetchnewseq";
    const response = await fetch(url, {
      method: 'POST',
      headers: {
        'X-CSRFToken': csrftoken,
        'Content-Type': 'application/json'
      },
      body: JSON.stringify(comandi_json)
    });
    return await response.json();
  }

  postData().then((data) => {
    window.location.replace(data.url);
  });
}

HTML按钮调用:

<button onclick="post_db()" class="btn btn-success">确认</button>

后端Django视图:

def fetch_new_seq(request):
    json_ricevuto = json.loads(request.body.decode("utf-8"))
    print(json_ricevuto) # 可能会在此处看到两次打印
    messages.success(request, "收到")
    redirect_url = reverse('newSeq')
    return JsonResponse({'url': redirect_url})

在这种设置下,如果后端打印了两次json_ricevuto,则表明Fetch请求被重复发送。同时,客户端浏览器控制台可能会出现Uncaught (in promise) TypeError: NetworkError when attempting to fetch resource.的错误。

问题根源分析:异步函数在循环中的不当使用

经过排查,Fetch请求重复发送的根本原因往往在于异步请求逻辑被错误地放置在一个循环内部。当循环的每次迭代都满足某个条件时,Fetch请求就会被重复触发。

Tome
Tome

先进的AI智能PPT制作工具

下载

以下是一个导致Fetch重复触发的典型错误代码结构:

function post_db() {
  // ... 其他代码
  for (var i = 0; i < len_array; i++) {
    if (
      comandi_json["lista_comandi"][i]["comando"] == "" ||
      comandi_json["lista_comandi"][i]["tempo"] == null
    ) {
      console.log("Error: 校验失败");
      return; // 如果校验失败,则提前退出函数
    } else {
      // 错误示范:将异步请求函数定义并调用在循环内部的else块中
      async function postData() {
        const url = "fetchnewseq";
        const response = await fetch(url, {
          method: "POST",
          headers: {
            "X-CSRFToken": csrftoken,
            "Content-Type": "application/json",
          },
          body: JSON.stringify(comandi_json),
        });
        return await response.json();
      }
      postData().then((data) => {
        window.location.replace(data.url);
      });
    }
  }
}

在这个错误示例中,postData 异步函数及其调用被放置在 for 循环的 else 块中。这意味着,只要 comandi_json["lista_comandi"][i] 中的某个元素通过了校验(即 if 条件不满足),else 块就会执行,从而触发一次新的Fetch请求。如果 len_array 大于1,并且有多个元素通过了校验,那么Fetch请求就会被发送多次。

至于NetworkError,它很可能是在第一个成功的Fetch请求完成并执行window.location.replace(data.url)后发生的。页面重定向会导致浏览器终止所有当前正在进行或即将进行的网络请求。因此,如果循环触发了多个Fetch请求,第一个请求成功后页面重定向,后续的Fetch请求就会因页面卸载而被中止,从而抛出NetworkError。

解决方案:分离校验逻辑与异步请求

解决此问题的关键在于将数据校验逻辑与异步网络请求逻辑清晰地分离。循环应该只负责完成所有必要的校验,而Fetch请求则应该在所有校验通过后,作为一次性操作来执行。

以下是修正后的代码结构:

function post_db() {
  // ... 其他代码

  // 步骤1:首先,完成所有的数据校验。
  // 循环只用于检查数据完整性,不触发任何网络请求。
  for (var i = 0; i < len_array; i++) {
    if (
      comandi_json["lista_comandi"][i]["comando"] == "" ||
      comandi_json["lista_comandi"][i]["tempo"] == null
    ) {
      console.log("Error: 校验失败,缺少必要数据。");
      // 如果任何一个元素校验失败,则立即退出函数,不发送请求。
      return; 
    }
  }

  // 步骤2:如果所有校验都通过,则在循环外部定义并执行异步请求。
  // 确保Fetch请求只被调用一次。
  async function postData() {
    const url = "fetchnewseq";
    const response = await fetch(url, {
      method: "POST",
      headers: {
        "X-CSRFToken": csrftoken,
        "Content-Type": "application/json",
      },
      body: JSON.stringify(comandi_json),
    });
    // 检查响应状态码,处理可能的HTTP错误
    if (!response.ok) {
        throw new Error(`HTTP error! status: ${response.status}`);
    }
    return await response.json();
  }

  // 步骤3:执行异步请求,并处理响应。
  postData()
    .then((data) => {
      // 请求成功后,进行页面重定向或其他操作
      window.location.replace(data.url);
    })
    .catch((error) => {
      // 捕获并处理Fetch请求过程中可能发生的错误
      console.error("Fetch请求失败:", error);
      // 可以显示用户友好的错误消息
    });
}

通过将postData函数的定义和调用移到循环之外,我们确保了:

  1. 校验先行: 循环首先完成所有必要的输入数据校验。
  2. 单次请求: 只有当所有数据都通过校验后,postData函数才会被调用一次,从而发送一次Fetch请求。
  3. 避免重复: 彻底杜绝了Fetch请求被多次触发的问题。

总结与最佳实践

  • 分离职责: 在编写涉及异步操作和数据校验的代码时,始终牢记分离职责的原则。循环应专注于数据处理或校验,而异步请求应作为独立的操作,在满足所有前置条件后执行。
  • 异步操作的生命周期: 理解fetch等异步操作的生命周期。当页面发生重定向或卸载时,正在进行的网络请求可能会被中断,导致NetworkError。
  • 调试技巧: 遇到类似问题时,利用浏览器的开发者工具(Network Tab)是定位问题的关键。它可以清晰地显示所有发出的网络请求,包括请求的次数、状态和响应。同时,在代码中合理使用console.log进行日志输出,可以帮助追踪代码执行路径。
  • 错误处理: 对于Fetch请求,始终添加.catch()块来捕获和处理可能发生的网络错误或其他异常,提高应用的健壮性。

遵循这些最佳实践,可以有效避免Fetch请求重复发送的问题,构建更稳定、高效的Web应用程序。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Python Web 框架 Django 深度开发
Python Web 框架 Django 深度开发

本专题系统讲解 Python Django 框架的核心功能与进阶开发技巧,包括 Django 项目结构、数据库模型与迁移、视图与模板渲染、表单与认证管理、RESTful API 开发、Django 中间件与缓存优化、部署与性能调优。通过实战案例,帮助学习者掌握 使用 Django 快速构建功能全面的 Web 应用与全栈开发能力。

166

2026.02.04

resource是什么文件
resource是什么文件

Resource文件是一种特殊类型的文件,它通常用于存储应用程序或操作系统中的各种资源信息。它们在应用程序开发中起着关键作用,并在跨平台开发和国际化方面提供支持。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

181

2023.12.20

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

847

2023.08.22

console接口是干嘛的
console接口是干嘛的

console接口是一种用于在计算机命令行或浏览器开发工具中输出信息的工具,提供了一种简单的方式来记录和查看应用程序的输出结果和调试信息。本专题为大家提供console接口相关的各种文章、以及下载和课程。

420

2023.08.08

console.log是什么
console.log是什么

console.log 是 javascript 函数,用于在浏览器控制台中输出信息,便于调试和故障排除。想了解更多console.log的相关内容,可以阅读本专题下面的文章。

541

2024.05.29

promise的用法
promise的用法

“promise” 是一种用于处理异步操作的编程概念,它可以用来表示一个异步操作的最终结果。Promise 对象有三种状态:pending(进行中)、fulfilled(已成功)和 rejected(已失败)。Promise的用法主要包括构造函数、实例方法(then、catch、finally)和状态转换。

337

2023.10.12

html文本框类型介绍
html文本框类型介绍

html文本框类型有单行文本框、密码文本框、数字文本框、日期文本框、时间文本框、文件上传文本框、多行文本框等等。详细介绍:1、单行文本框是最常见的文本框类型,用于接受单行文本输入,用户可以在文本框中输入任意文本,例如用户名、密码、电子邮件地址等;2、密码文本框用于接受密码输入,用户在输入密码时,文本框中的内容会被隐藏,以保护用户的隐私;3、数字文本框等等。

429

2023.10.12

location.assign
location.assign

在前端开发中,我们经常需要使用JavaScript来控制页面的跳转和数据的传递。location.assign就是JavaScript中常用的一个跳转方法。通过location.assign,我们可以在当前窗口或者iframe中加载一个新的URL地址,并且可以保存旧页面的历史记录。php中文网为大家带来了location.assign的相关知识、以及相关文章等内容,供大家免费下载使用。

232

2023.06.27

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

1

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 6万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 3.4万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号