Spring Boot 异步编程深入剖析

Spring Boot 异步编程深入剖析

1. 异步方法的使用
原理深度解析

Spring Boot 的异步方法基于 Spring 的 AOP(面向切面编程)实现。当在方法上添加 @Async 注解时,Spring 会为该方法所在的类创建一个代理对象。当调用该异步方法时,实际上是调用代理对象的方法,代理对象会将该方法的执行委托给线程池中的一个线程去执行,而调用线程会继续执行后续代码,从而实现异步执行。

更复杂的使用场景

除了返回 CompletableFuture,还可以使用 ListenableFuture(在 Spring 4.0 之前)或无返回值的异步方法。

import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
import org.springframework.util.concurrent.ListenableFuture;
import org.springframework.util.concurrent.SettableListenableFuture;import java.util.concurrent.CompletableFuture;@Service
public class AsyncService {// 无返回值的异步方法@Asyncpublic void asyncVoidMethod() {try {Thread.sleep(2000);System.out.println("Async void method completed");} catch (InterruptedException e) {e.printStackTrace();}}// 使用 ListenableFuture@Asyncpublic ListenableFuture<String> asyncListenableMethod() {SettableListenableFuture<String> future = new SettableListenableFuture<>();try {Thread.sleep(2000);future.set("Async listenable method completed");} catch (InterruptedException e) {future.setException(e);}return future;}
}
踩坑记录
  • 方法调用问题:如果在同一个类中调用自身的异步方法,@Async 注解不会生效。因为 Spring 的 AOP 代理是基于外部调用的,同一个类中的方法调用不会经过代理对象。解决方法是将异步方法提取到另一个服务类中。
  • 异常处理问题:无返回值的异步方法中的异常不会被调用者捕获,因为调用者不会等待方法执行完成。可以在异步方法内部进行异常处理,或者使用 CompletableFuture 来捕获异常。
使用心得
  • 对于一些耗时的 I/O 操作(如数据库查询、网络请求等),使用异步方法可以显著提高应用程序的响应性能。
  • 合理使用 CompletableFuture 可以方便地处理异步任务的结果和异常,同时还可以进行任务的组合和链式调用。
2. 线程池配置
深入理解线程池参数
  • 核心线程数(corePoolSize:线程池保持的最小线程数。当有新任务提交时,如果线程池中的线程数小于核心线程数,会创建新的线程来执行任务。
  • 最大线程数(maxPoolSize:线程池允许的最大线程数。当队列已满且线程数小于最大线程数时,会创建新的线程来执行任务。
  • 队列容量(queueCapacity:用于存储等待执行的任务的队列的容量。当线程池中的线程数达到核心线程数时,新任务会被放入队列中等待执行。
  • 线程空闲时间(keepAliveTime:当线程池中的线程数超过核心线程数时,空闲线程在经过一定时间后会被销毁。
动态调整线程池参数

可以通过编写自定义的线程池管理器来动态调整线程池的参数。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.AsyncConfigurer;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;import java.util.concurrent.Executor;@Configuration
public class AsyncConfig implements AsyncConfigurer {private ThreadPoolTaskExecutor executor;@Override@Bean(name = "asyncExecutor")public Executor getAsyncExecutor() {executor = new ThreadPoolTaskExecutor();executor.setCorePoolSize(5);executor.setMaxPoolSize(10);executor.setQueueCapacity(25);executor.setThreadNamePrefix("AsyncThread-");executor.initialize();return executor;}public void adjustCorePoolSize(int corePoolSize) {executor.setCorePoolSize(corePoolSize);executor.initialize();}
}
踩坑记录
  • 队列容量设置不合理:如果队列容量设置过大,可能会导致大量任务堆积在队列中,从而影响系统的响应性能。如果队列容量设置过小,可能会导致线程池频繁创建新的线程,增加系统的开销。
  • 线程池耗尽问题:如果任务提交速度过快,超过了线程池的处理能力,可能会导致线程池耗尽,从而抛出 RejectedExecutionException 异常。可以通过合理设置线程池参数和实现自定义的拒绝策略来解决这个问题。
使用心得
  • 根据应用程序的实际情况合理设置线程池的参数,避免资源浪费和性能瓶颈。
  • 定期监控线程池的状态,根据系统的负载情况动态调整线程池的参数。
3. 异步任务的监控与管理
高级监控方法

除了使用 CompletableFuture 来监控任务状态,还可以使用 Spring Boot Actuator 来监控线程池的状态。

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-actuator</artifactId>
</dependency>

通过访问 /actuator/metrics 端点可以查看线程池的相关指标,如活跃线程数、任务完成数等。

任务链管理

可以使用 CompletableFuturethenApplythenAcceptthenCompose 等方法来构建任务链,实现复杂的异步任务管理。

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;import java.util.concurrent.CompletableFuture;@RestController
public class AsyncController {@Autowiredprivate AsyncService asyncService;@GetMapping("/asyncChain")public String asyncChain() {CompletableFuture<String> future1 = asyncService.asyncMethod();CompletableFuture<String> future2 = future1.thenApply(result -> result + " -> Next step");future2.thenAccept(finalResult -> System.out.println(finalResult));return "Async chain started";}
}
踩坑记录
  • 任务链异常处理问题:在任务链中,如果某个任务抛出异常,后续的任务可能会受到影响。可以使用 exceptionally 方法来处理异常,确保任务链的稳定性。
  • 内存泄漏问题:如果 CompletableFuture 没有正确处理,可能会导致内存泄漏。例如,如果一个 CompletableFuture 一直处于未完成状态,会占用内存资源。
使用心得
  • 利用任务链可以实现复杂的异步业务逻辑,提高代码的可读性和可维护性。
  • 及时处理异步任务中的异常,避免异常扩散导致系统崩溃。

通过深入理解 Spring Boot 异步编程的原理和机制,合理配置线程池,以及有效地监控和管理异步任务,可以充分发挥异步编程的优势,提高应用程序的性能和响应能力。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/27593.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

简易的微信聊天网页版【项目测试报告】

文章目录 一、项目背景二、项目简介登录功能好友列表页面好友会话页面 三、测试工具和环境四、测试计划测试用例部分人工手动测试截图web自动化测试测试用例代码框架配置内容代码文件&#xff08;Utils.py&#xff09;登录页面代码文件&#xff08;WeChatLogin.py&#xff09;好…

QT播放视频保持视频宽高比消除黑边

QT播放视频保持视频宽高比消除黑边 1、问题 在播放视频的时候&#xff0c;由于框架的大小发生变化&#xff0c;导致视频出现黑边很不好看。 因此需要像一种方法消除黑边 2、处理 1、读取视频的宽高比 2、设置视频的Widget的大小固定&#xff0c;Widget的宽高比和视频宽高比…

Apache nifi demo 实验

Apache nifi 是个数据流系统&#xff0c;可以通过配置 自定义的流程来实现数据的转换。 比如可以配置一个流程&#xff0c;读取数据库里的数据&#xff0c;再转换&#xff0c;最后保存到本地文件。 这样可以来实现一些数据转换的操作&#xff0c;而不用特地编写程序来导入导出。…

蓝桥杯 - 每日打卡(类斐波那契循环数)

题目: 解题思路&#xff1a; 假设输入数值为number 分析题目&#xff0c;如果想要解决这个问题&#xff0c;我们需要实现两个方法&#xff0c;第一个检查number是否是类斐波那契&#xff0c;第二个是模拟1e7 - 0的过程&#xff0c;因为是求最大的&#xff0c;那么我们从1e7开始…

DeepSeek集成到VScode工具,让编程更高效

DeepSeek与VScode的强强联合&#xff0c;为编程效率树立了新标杆。 DeepSeek&#xff0c;一款卓越的代码搜索引擎&#xff0c;以其精准的索引和高速的检索能力&#xff0c;助力开发者在浩瀚的代码海洋中迅速定位关键信息。 集成至VScode后&#xff0c;开发者无需离开熟悉的编辑…

RAG组件:向量数据库(Milvus)

在当前大模型盛行的时代&#xff0c;大模型的垂类微调、优化成为产业落地、行业应用的关键&#xff1b;RAG技术应运而生&#xff0c;主要解决大模型对专业知识、实效性知识欠缺的问题&#xff1b; RAG的核心工作逻辑是将专业知识、实效知识等大模型欠缺的知识进行收集、打包、保…

2022java面试总结,1000道(集合+JVM+并发编程+Spring+Mybatis)的Java高频面试题

1、面试题模块汇总 面试题包括以下十九个模块&#xff1a; Java 基础、容器、多线程、反射、对象拷贝、Java Web 模块、异常、网络、设计模式、Spring/Spring MVC、Spring Boot/Spring Cloud、Hibernate、Mybatis、RabbitMQ、Kafka、Zookeeper、MySql、Redis、JVM 。如下图所示…

英码科技携昇腾DeepSeek大模型一体机亮相第三届北京人工智能产业创新发展大会

2025年2月28日&#xff0c;第三届北京人工智能产业创新发展大会在国家会议中心隆重开幕。本届大会以"好用、易用、愿用——以突破性创新加速AI赋能千行百业”为主题&#xff0c;重点展示人工智能技术创新成果与产业化应用实践。作为昇腾生态的APN伙伴&#xff0c;英码科技…

(链表 删除链表的倒数第N个结点)leetcode 19

设空结点指向head便于插入和删除结点 考虑特殊情况 head结点被删除 a结点仅用来测试长度&#xff0c;找到目标结点的位置 b结点为空结点指向head返回值 cur用来删除目标值&#xff08;特殊情况 目标值为head 这时curb) 则开始就将cur初始化为b开始遍历 /*** Definition fo…

Android Studio 新版本Gradle发布本地Maven仓库示例

发布代码到JitPack示例&#xff1a;https://blog.csdn.net/loutengyuan/article/details/145938967 以下是基于 Android Studio 24.2.2&#xff08;Gradle 8.10.2 AGP 8.8.0 JDK17&#xff09; 的本地 Maven 仓库发布示例&#xff0c;包含aar和jar的不同配置&#xff1a; 1.…

[今年毕业设计]最新最全最有创意的基于云计算的计算机专业毕设选题精选推荐汇总建议收藏!!

文章目录 前言基于云计算的毕业设计选题毕设作品展示 前言 2025全新毕业设计项目 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云等平台优质作者。 &#x1f447;&#x1f3fb; 精彩专栏 推荐订阅&#x1f4…

coze生成的工作流,发布后,利用cmd命令行执行。可以定时发日报,周报等。让他总结你飞书里面的表格。都可以

coze生成的工作流&#xff0c;发布后&#xff0c;利用cmd命令行执行。可以定时发日报&#xff0c;周报等。让他总结你飞书里面的表格。都可以。 很简单。 准备工作&#xff0c;先发布你的工作流&#xff0c;和发布应用。 然后&#xff0c;点击扣子API 。 申请一个&#xff0…

安防监控/视频集中存储EasyCVR视频汇聚平台如何配置AI智能分析平台的接入?

EasyCVR安防视频监控平台不仅支持AI边缘计算智能硬件设备的接入&#xff0c;还能快速集成AI智能分析平台&#xff0c;接收来自智能分析平台或设备的AI告警信息&#xff0c;如烟火检测、周界入侵检测、危险区域闯入检测、安全帽/反光衣佩戴检测等。 本文将详细介绍如何在EasyCVR…

以1.7K深圳小区房价为例,浙大GIS实验室使用注意力机制挖掘地理情景特征,提升空间非平稳回归精度

地理加权回归 (Geographically Weighted Regression, GWR) 是一种广泛应用于地理空间分析的统计方法&#xff0c;用于捕捉地理现象的空间非平稳性&#xff08;即空间异质性&#xff09;。传统 GWR 通过为每个观测点分配权重来反映其对回归参数的影响&#xff0c;这些权重通常基…

【JavaEE】线程安全

【JavaEE】线程安全 一、引出线程安全二、引发线程安全的原因三、解决线程安全问题3.1 synchronized关键字&#xff08;解决修改操作不是原子的&#xff09;3.1.1 synchronized的特性3.1.1 synchronized的使用事例 3.2 volatile 关键字&#xff08;解决内存可见性&#xff09; …

智慧农业中光谱相机对土壤成分的无损检测应用‌

可浏览之前发布的一篇文章&#xff1a;光谱相机在农业中的具体应用案例 一、土壤成分定量分析 ‌养分检测‌ 光谱相机通过捕捉土壤反射的特定波长光线&#xff0c;可精准检测氮、磷、钾等主要养分含量&#xff0c;以及有机质和水分比例。例如&#xff0c;不同养分对近红外波段…

实现浏览器交互Ai Web Ui-本地化部署的deepseek + Ollama + Page Assist

一、deepseek本地化部署 上篇写了deepseek本地化部署的小白教程。 deepseek 本地化部署&#xff08;小白也可部署&#xff09; 但有个问题&#xff0c;Ollama只能在命令行进行交互&#xff0c;体验相当差。 二、Page Assist安装 本篇给大家介绍个好用的chrome浏览器AI Web …

使用Maven搭建Spring Boot框架

文章目录 前言1.环境准备2.创建SpringBoot项目3.配置Maven3.1 pom.xml文件3.2 添加其他依赖 4. 编写代码4.1 启动类4.2 控制器4.3 配置文件 5.运行项目6.打包与部署6.1 打包6.2 运行JAR文件 7.总结 前言 Spring Boot 是一个用于快速构建 Spring 应用程序的框架&#xff0c;它简…

易语言模拟真人鼠标轨迹算法 - 防止游戏检测

一.简介 鼠标轨迹算法是一种模拟人类鼠标操作的程序&#xff0c;它能够模拟出自然而真实的鼠标移动路径。 鼠标轨迹算法的底层实现采用C/C语言&#xff0c;原因在于C/C提供了高性能的执行能力和直接访问操作系统底层资源的能力。 鼠标轨迹算法具有以下优势&#xff1a; 模拟…

10.【线性代数】—— 四个基本子空间

十、 四个基本子空间 1. 列空间 C ( A ) C(A) C(A) in R m R^m Rm2. 零空间 N ( A ) N(A) N(A) in R n R^n Rn3. 行空间 C ( A T ) C(A^T) C(AT) in R n R^n Rn4. 左零空间 N ( A T ) N(A^T) N(AT) in R m R^m Rm综述5. 新的向量空间 讨论矩阵 A m ∗ n A_{m*n} Am∗n​…