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

springboot中的异步任务

在springboot项目中可以通过@EnableAsync+@Async的方式简化异步操作,下文使用springboot:3.2.1 源码分析

若一个bean中的公共方法上标注了@Async,在系统启动时,会给这个类创建一个代理对象,并将该代理对象作为bean注册到spring容器中
当调用带有@Async注解的方法时,实际上是调用了代理对象的方法,在代理对象的方法中将真正的方法交给线程池去执行

原理分析

代理方法执行时,AsyncExecutionInterceptor#invoke调用determineAsyncExecutor()方法获取异步执行器,然后将真实方法的执行交给异步执行器
在这里插入图片描述
获取@Async指定的异步执行器,若未指定,则使用默认的异步执行器
在这里插入图片描述

获取默认的异步执行器

  • 默认的异步执行器类型是ThreadPoolTaskExecutor,由org.springframework.boot.autoconfigure.task.TaskExecutorConfigurations.TaskExecutorConfiguration向spring容器中注册
  • 且核心线程数默认是8,阻塞队列容量默认是Integer.MAX_VALUE,可见默认的设置在生产环境使用时存在资源耗尽的风险。可通过配置文件调整(配置项对应类:org.springframework.boot.autoconfigure.task.TaskExecutionProperties)

如果AsyncConfigurer提供了Executor,则使用该Executor作为默认执行器,否则通过getDefaultExecutor方法获取默认执行器
在这里插入图片描述
AsyncExecutionInterceptor#getDefaultExecutor方法中,首先从父类方法中获取执行器(从spring容器中获取org.springframework.core.task.TaskExecutor类型的bean或者name是taskExecutor的bean),若为空,则使用SimpleAsyncTaskExecutor
在这里插入图片描述
在这里插入图片描述
默认情况下,spring容器中并没有TaskExecutor类型的bean,但是在TaskExecutorConfiguration(该配置类是由TaskExecutionAutoConfiguration自动配置类使用@Import导入)配置类中,向spring容器中注册了name是taskExecutor的bean
在这里插入图片描述

自定义异步执行器

自定义默认异步执行器

以下两种方式都是在自定义默认的异步执行器,也就说@Async不指定线程池时使用的默认异步执行器

  1. 自定义配置类,并实现org.springframework.scheduling.annotation.AsyncConfigurer接口,重写getAsyncExecutor方法
  2. 向容器中注册TaskExecutor类型的bean,覆盖内置的TaskExecutor
自定义普通异步执行器
  • 自定义普通异步执行器的bean名称,不可设置为taskExecutor,且类型不可以是TaskExecutor,否则会成为默认异步执行器
  • 可创建org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor对象,并自定义其属性值,然后将该对象注册到spring容器中

异常处理

异步方法的异常捕获和处理
在这里插入图片描述
在这里插入图片描述
默认的异常处理器SimpleAsyncUncaughtExceptionHandler
在这里插入图片描述
在这里插入图片描述
示例

2024-09-20T15:59:20.776+08:00 ERROR 34764 --- [         task-1] .a.i.SimpleAsyncUncaughtExceptionHandler : Unexpected exception occurred invoking async method: public void com.example.box.dynamicproxy.TestJdkProxyServiceImpl.run()java.lang.RuntimeException: 公司又停发工资了!!!at com.example.box.dynamicproxy.TestJdkProxyServiceImpl.run(TestJdkProxyServiceImpl.java:14) ~[classes/:na]at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) ~[na:na]at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:77) ~[na:na]

自定义异常处理器

配置类实现AsyncConfigurer接口的getAsyncUncaughtExceptionHandler()方法,返回一个自定义的AsyncUncaughtExceptionHandler实现类的实例
在这里插入图片描述

http://www.lryc.cn/news/444946.html

相关文章:

  • Linux学习笔记8 理解Ubuntu网络管理,做自己网络的主人
  • 理解线程的三大特性:原子性、可见性和有序性
  • 英特尔®以太网网络适配器E810-CQDA1 / E810-CQDA2 网卡 规格书 e810 网卡 规格书 Intel100G E810 网卡 白皮书
  • 好用的idea方法分隔符插件
  • 通过 Xshell 无法连接到 Ubuntu
  • Java面试篇基础部分-Synchronized关键字详解
  • 数据结构之线性表——LeetCode:67. 二进制求和,27. 移除元素,26. 删除有序数组中的重复项
  • SQL_HAVING小例子
  • Avalonia第三方UI库Semi.Avalonia用法详解
  • 宠物智能化听诊器的健康管理!
  • MyBatis-Plus 实体类注解
  • 如何写一个自动化Linux脚本去进行等保测试--引言
  • 美团测开OC!
  • HyperWorks的实体几何创建与六面体网格剖分
  • 项目实战:Ingress搭建Nginx+WP论坛+MariaDB
  • UWA支持鸿蒙HarmonyOS NEXT
  • 【齐家网-注册/登录安全分析报告】
  • MyBatis 基本概念
  • 前端开发之装饰器模式
  • 【STL】pair 与 map:基础、操作与应用
  • 深度学习-图像处理篇4VGG网络
  • 初级css+初级选择器
  • gitlab 的CI/CD (二)
  • 【html】基础(一)
  • 【网站架构部署与优化】Nginx优化
  • gitlab修改访问端口
  • 分库分表-分页排序查询
  • 【openwrt-21.02】openwrt PPTP Passthrough 不生效问题解决方案
  • 【编程基础知识】Mysql的各个索引数据结构及其适用场景
  • 解决IDEA出现:java: 程序包javax.servlet不存在的问题