当前位置: 首页 > news >正文

前端Vue3 + 后端Spring Boot,前端取消请求后端处理逻辑分析

在 Vue3 + Spring Boot 的技术栈下,前端取消请求后,后端是否继续执行业务逻辑的答案仍然是 取决于请求处理的阶段 和 Spring Boot 的实现方式。以下是结合具体技术的详细分析:


1. 请求未到达 Spring Boot

  • 场景:前端通过 AbortController 取消请求(如 axios 的 signal),但请求尚未到达服务器。

  • 结果:Spring Boot 完全不会收到该请求,业务逻辑不会执行。


2. 请求已到达 Spring Boot

  • 场景:请求已进入 Spring Boot 的 Controller 层并开始处理。

  • 结果默认情况下,Spring Boot 会继续执行完业务逻辑。原因:

    1. HTTP 协议特性:Spring Boot 无法主动感知客户端是否断开连接(除非显式监听)。

    2. 线程模型:Spring Boot 默认使用线程池处理请求,一旦任务提交到线程池,即使客户端断开,线程仍会继续执行任务。

  • 示例代码

    java

    复制

    下载

    @PostMapping("/submit")
    public ResponseEntity<String> submitData(@RequestBody Data data) {// 假设这是一个耗时操作(如数据库写入)someLongRunningService.process(data);return ResponseEntity.ok("Success");
    }

    即使前端取消请求,someLongRunningService.process(data) 仍会执行完毕。


3. 如何让 Spring Boot 终止处理?

  • 需要显式监听客户端断开事件,结合 Spring Boot 的异步机制实现:

方案 1:使用 DeferredResult 监听连接状态

复制

下载

 ```java@PostMapping("/submit")public DeferredResult<String> submitData(@RequestBody Data data) {DeferredResult<String> deferredResult = new DeferredResult<>();// 监听客户端断开事件deferredResult.onTimeout(() -> {// 执行终止逻辑(如关闭数据库连接、中断线程等)cleanup();});// 异步执行业务逻辑CompletableFuture.runAsync(() -> {someLongRunningService.process(data);deferredResult.setResult("Success");});return deferredResult;}```- **优点**:通过 `DeferredResult.onTimeout()` 或 `onCompletion()` 监听客户端断开。- **缺点**:需要手动终止异步任务(如调用线程中断)。
方案 2:结合响应式编程(WebFlux)

复制

下载

 如果使用 **Spring WebFlux**(非阻塞异步模型),可通过响应式流控制中断:```java@PostMapping("/submit")public Mono<String> submitData(@RequestBody Data data) {return Mono.fromCallable(() -> someLongRunningService.process(data)).timeout(Duration.ofSeconds(30)) // 设置超时.doOnCancel(() -> cleanup());    // 监听取消事件}```- **优点**:天然支持非阻塞中断。- **缺点**:需要重构为响应式代码。
方案 3:自定义线程中断

复制

下载

 在业务逻辑中检查线程中断状态:```javapublic void process(Data data) {while (!Thread.currentThread().isInterrupted()) {// 执行可中断的任务}}```在客户端断开时,调用 `Thread.interrupt()` 终止任务(需结合 `DeferredResult` 使用)。

4. 关键注意事项

  • 资源泄漏风险
    如果客户端断开后未正确终止数据库连接、文件句柄等资源,可能导致资源泄漏。

  • 数据一致性
    对于支付、订单等关键操作,即使前端取消,后端可能已完成处理。需通过以下方式保证一致性:

    1. 幂等性设计:通过唯一请求 ID 避免重复处理。

    2. 状态查询接口:前端取消后,轮询后端状态确认是否成功。

    3. 事务回滚:在监听到客户端断开时,手动回滚事务(需结合 @Transactional)。


5. 总结

场景是否继续执行解决方案建议
请求未到达 Spring Boot无需处理
请求到达且正在处理(默认)使用 DeferredResult 或 WebFlux
请求到达且已监听客户端断开可终止显式中断线程或清理资源

推荐方案

  • 对耗时操作(如文件上传、复杂计算),使用 DeferredResult + 线程中断 实现可终止逻辑。

  • 对关键业务(如支付),通过 幂等性 + 状态查询 确保数据一致性。

实例调试代码:

AsyncConfig.java

package com.weiyu.config;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;@Configuration
public class AsyncConfig {@Bean(name = "asyncTaskExecutor") // 明确指定Bean名称public ThreadPoolTaskExecutor asyncExecutor() {ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();// 核心线程数 = CPU核心数 + 1executor.setCorePoolSize(Runtime.getRuntime().availableProcessors() + 1);executor.setMaxPoolSize(50);executor.setQueueCapacity(100);executor.setThreadNamePrefix("Async-QueryTask-");executor.initialize(); // 必须初始化!!!return executor;}
}
TaskMessageController.java
// 正确注入方式@Qualifier("asyncTaskExecutor")@Autowiredprivate Executor asyncTaskExecutor;@GetMapping("/task")public DeferredResult<Result<List<Task>>> queryTask(){DeferredResult<Result<List<Task>>> deferredResult = new DeferredResult<>();// 监听客户端断开事件deferredResult.onTimeout(() -> {deferredResult.setErrorResult(Result.error("请求超时"));});// 使用自定义线程池(推荐)或公共池执行异步任务CompletableFuture.supplyAsync(() -> {// 这里执行实际业务逻辑List<Task> taskList = taskMessageService.queryTask();// 防御性空值检查if (taskList == null) {return Result.error("服务返回空结果");}return Result.success(taskList);}, asyncTaskExecutor).whenComplete((result, ex) -> {if (ex != null) {// 异常处理deferredResult.setErrorResult(Result.error(ex.getMessage()));} else {// 正常返回结果deferredResult.setResult(result);}});return deferredResult;}

 

相关文章:

  • 【MySQL】(9) 视图
  • 跟我学C++中级篇——控制死锁
  • Qt开发:JSON字符串的序列化和反序列化
  • 【OSG学习笔记】Day 14: 操作器(Manipulator)的深度使用
  • 基于机器学习的电影票房预测
  • 万象生鲜配送系统代码2025年4月29日更新日志
  • LeetCode 155题解 | 最小栈
  • 【Leetcode 每日一题 - 补卡】2302. 统计得分小于 K 的子数组数目
  • Linux电源管理(3)_关机和重启的过程
  • 第十六届蓝桥杯 2025 C/C++组 密密摆放
  • 探索语音增强中的多尺度时间频率卷积网络(TFCM):代码解析与概念介绍
  • AI赋能的问答系统:2025年API接口实战技巧
  • 【Redis——数据类型和内部编码和Redis使用单线程模型的分析】
  • 基于Arduino的STM32F103RCT6最小系统板的测试及串口通讯
  • 深度学习中的优化算法:基础全面解析
  • 聊聊Spring AI Alibaba的PlantUMLGenerator
  • 安装deepspeed时出现了以下的错误,如何解决CUDA_HOME does not exist
  • 【Java面试笔记:进阶】28.谈谈你的GC调优思路?
  • 解决STM32H743单片机USB_HOST+FATF操作usb文件
  • 从 Pretrain 到 Fine-tuning:大模型迁移学习的核心原理剖析
  • 从腰缠万贯到债台高筑、官司缠身:尼泊尔保皇新星即将陨落?
  • 江苏银行一季度净赚近98亿增逾8%,不良贷款率微降
  • 马上评|“AI神医宇宙”欺诈,连演员都不请了
  • 广州海关原党委委员、副关长刘小威被开除党籍
  • 北上广深还是小城之春?“五一”想好去哪玩了吗
  • 民调显示特朗普执政百日支持率为80年来美历任总统最低