SpringBoot 中的异步处理机制详解

👉 这是一个或许对你有用的社群

🐱 一对一交流/面试小册/简历优化/求职解惑,欢迎加入「芋道快速开发平台」知识星球。下面是星球提供的部分资料: 

b0b10d7043b5c35dae7462d4f6edac80.gif

👉这是一个或许对你有用的开源项目

国产 Star 破 10w+ 的开源项目,前端包括管理后台 + 微信小程序,后端支持单体和微服务架构。

功能涵盖 RBAC 权限、SaaS 多租户、数据权限、商城、支付、工作流、大屏报表、微信公众号、ERP、CRMAI 大模型等等功能:

  • Boot 多模块架构:https://gitee.com/zhijiantianya/ruoyi-vue-pro

  • Cloud 微服务架构:https://gitee.com/zhijiantianya/yudao-cloud

  • 视频教程:https://doc.iocoder.cn

【国内首批】支持 JDK 17/21 + SpringBoot 3.3、JDK 8/11 + Spring Boot 2.7 双版本 

来源:juejin.cn/post/
7424693750477848614


在现代 Web 应用程序开发中,处理并发任务和提高应用程序的性能是一个重要的主题。Spring Boot 提供了强大的异步处理机制,允许我们通过简单的配置实现多线程任务处理,避免阻塞主线程,从而提升应用的响应速度和吞吐量。本文将深入探讨 Spring Boot 中的异步处理机制,介绍其原理、配置方法以及实际使用场景。

1、为什么需要异步处理?

在 Web 应用程序中,某些任务可能需要花费较长的时间,比如调用外部服务、执行文件 I/O 操作或处理复杂的计算逻辑。如果这些任务在主线程(即请求处理线程)中执行,会导致响应变慢,影响用户体验。通过异步处理,我们可以将这些耗时任务交给后台线程执行,释放主线程,尽早返回响应。

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/ruoyi-vue-pro

  • 视频教程:https://doc.iocoder.cn/video/

2、Spring Boot 异步处理的基本原理

Spring Boot 中的异步处理基于 Spring Framework 的 @Async 注解,它利用线程池来异步执行任务。通过简单地在方法上加上 @Async,Spring 会自动在后台线程中执行该方法,而不会阻塞主线程。

Spring 的异步支持核心依赖以下两个组件:

  • @EnableAsync:启用异步处理的注解。

  • @Async:标注需要异步执行的方法。

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/yudao-cloud

  • 视频教程:https://doc.iocoder.cn/video/

3、Spring Boot 异步处理的配置

3.1. 启用异步支持

在使用异步功能之前,我们需要在 Spring Boot 应用的启动类或配置类中使用 @EnableAsync 注解启用异步支持:

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableAsync;

@SpringBootApplication
@EnableAsync
public class AsyncApplication {
    public static void main(String[] args) {
        SpringApplication.run(AsyncApplication.class, args);
    }
}
3.2. 创建异步方法

接下来,我们可以在服务层创建一个异步执行的方法。使用 @Async 注解标注的方法会在单独的线程中执行。

import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;

@Service
public class AsyncService {

    @Async
    public void executeAsyncTask() {
        System.out.println("执行异步任务: " + Thread.currentThread().getName());
        // 模拟长时间任务
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("任务完成");
    }
}

在该例子中,executeAsyncTask() 方法被标记为异步执行,当它被调用时,将不会阻塞调用者线程。

3.3. 调用异步方法

我们可以在控制器或服务中调用异步方法:

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class AsyncController {

    private final AsyncService asyncService;

    public AsyncController(AsyncService asyncService) {
        this.asyncService = asyncService;
    }

    @GetMapping("/async")
    public String triggerAsyncTask() {
        asyncService.executeAsyncTask();
        return "异步任务已触发";
    }
}

当用户访问 /async 端点时,AsyncService 的异步任务会被触发,并立即返回响应,主线程不会等待异步任务的完成。

4、自定义线程池

默认情况下,Spring Boot 会使用一个简单的线程池来执行异步任务。但是在实际的生产环境中,我们通常需要对线程池进行更精细的配置。我们可以通过定义 Executor Bean 来自定义线程池。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.util.concurrent.Executor;

@Configuration
public class AsyncConfig {

    @Bean(name = "taskExecutor")
    public Executor taskExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(5);
        executor.setMaxPoolSize(10);
        executor.setQueueCapacity(25);
        executor.setThreadNamePrefix("Async-");
        executor.initialize();
        return executor;
    }
}

在这个配置中,我们创建了一个线程池 Executor,指定了核心线程数、最大线程数、队列容量等参数。线程池可以帮助我们更好地管理资源,避免大量并发任务导致系统资源耗尽。

为了让 Spring 使用这个自定义的线程池,我们需要在异步方法上指定线程池名称:

@Async("taskExecutor")
public void executeAsyncTask() {
    // 任务逻辑
}

5、异步方法的返回值

除了 void 类型,异步方法还可以返回 java.util.concurrent.Futureorg.springframework.util.concurrent.ListenableFuture,允许我们在异步任务完成后获取其结果。

例如,返回 Future 类型的异步方法:

import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Future;

@Async
public Future<String> asyncMethodWithReturn() {
    try {
        Thread.sleep(3000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    return CompletableFuture.completedFuture("异步任务结果");
}

调用该方法时,我们可以通过 Future.get() 获取异步任务的结果:

@GetMapping("/asyncResult")
public String getAsyncResult() throws Exception {
    Future<String> result = asyncService.asyncMethodWithReturn();
    return result.get();
}

需要注意的是,Future.get() 方法会阻塞调用线程,直到异步任务完成,因此在实际使用时我们应当尽量避免在主线程中直接调用 get()

6、异步处理的典型应用场景

  • 外部 API 调用:调用第三方服务时,响应时间可能较长,可以使用异步请求来避免阻塞主线程。

  • 文件上传/下载:处理大文件上传或下载时,异步任务可以提升用户体验,确保应用的其他功能不会因文件处理而变慢。

  • 消息队列:异步处理常用于消息队列中,比如接收和处理 Kafka、RabbitMQ 消息时,可以在后台异步处理消息,提升应用的并发能力。

7、总结

Spring Boot 的异步处理机制极大地简化了多线程开发,使得我们能够在无需管理复杂的线程逻辑的情况下,通过简单的注解实现异步任务执行。本文介绍了异步处理的基础配置、线程池的自定义以及常见应用场景。在实际应用中,异步处理可以有效提升应用的性能,改善用户体验,但同时也需要我们合理管理线程池,确保系统资源的高效利用。


欢迎加入我的知识星球,全面提升技术能力。

👉 加入方式,长按”或“扫描”下方二维码噢

021c1d1cc09edd4caab2e6da9db4eb24.png

星球的内容包括:项目实战、面试招聘、源码解析、学习路线。

1d4f0c10c5a2d348340aab8435e7b887.png

ab6bef0da73819e541e0c5d06ea922f3.pngc3772c5c2526765a2b600f3354bf1de5.pnga587aa813c4dc9dbcc8f21e5bb6a60b4.png7c409430d5d2025059ad642e581fa565.png

文章有帮助的话,在看,转发吧。
谢谢支持哟 (*^__^*)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值