当前位置:首页 > 科技  > 软件

浅析Spring中Async注解底层异步线程池原理

来源: 责编: 时间:2024-06-14 08:53:49 270观看
导读一、前言开发中我们经常会用到异步方法调用,具体到代码层面,异步方法调用的实现方式有很多种,比如最原始的通过实现Runnable接口或者继承Thread类创建异步线程,然后启动异步线程;再如,可以直接用java.util.concurrent包提供

一、前言

开发中我们经常会用到异步方法调用,具体到代码层面,异步方法调用的实现方式有很多种,比如最原始的通过实现Runnable接口或者继承Thread类创建异步线程,然后启动异步线程;再如,可以直接用java.util.concurrent包提供的线程池相关API实现异步方法调用。Odb28资讯网——每日最新资讯28at.com

如果说可以用一行代码快速实现异步方法调用,那是不是比上面方法香很多。Odb28资讯网——每日最新资讯28at.com

Spring提供了Async注解,就可以帮助我们一行代码搞定异步方法调用。Async注解用起来是很爽,但是如果不对其底层实现做深入研究,难免有时候也会心生疑虑,甚至会因使用不当,遇见一些让人摸不着头脑的问题。Odb28资讯网——每日最新资讯28at.com

本文首先将对Async注解做简单介绍,然后和大家分享一个我们项目中因Async注解使用不当的线上问题,接着再深扒Spring源码,对Async注解底层异步线程池的实现原理一探究竟。Odb28资讯网——每日最新资讯28at.com

二、Async注解简介

Odb28资讯网——每日最新资讯28at.com

Async注解定义源码

Odb28资讯网——每日最新资讯28at.com

图片图片Odb28资讯网——每日最新资讯28at.com

从源码可以看出@Async注解定义很简单,只需要关注两点:Odb28资讯网——每日最新资讯28at.com

  • Target({ElementType.TYPE, ElementType.METHOD})标志Async注解可以作用在方法和类上,作用在类上时,类的所有方法可以实现异步调用。
  • String value( ) default ""是唯一字段属性,用来指定异步线程池,且该字段有缺省值。

Odb28资讯网——每日最新资讯28at.com

Async注解异步调用实现原理概述Odb28资讯网——每日最新资讯28at.com

Odb28资讯网——每日最新资讯28at.com

在Spring框架中,Async注解的实现是通过AOP来实现的。具体来说,Async注解是由AsyncAnnotationAdvisor这个切面类来实现的。Odb28资讯网——每日最新资讯28at.com

AsyncAnnotationAdvisor类是Spring框架中用于处理Async注解的切面,它会在被Async注解标识的方法被调用时,创建一个异步代理对象来执行方法。这个异步代理对象会在一个新的线程中调用被@Async注解标识的方法,从而实现方法的异步执行。Odb28资讯网——每日最新资讯28at.com

在AsyncAnnotationAdvisor中,会使用AsyncExecutionInterceptor来处理Async注解。AsyncExecutionInterceptor是实现了MethodInterceptor接口的类,用于拦截被Async注解标识的方法的调用,并在一个新的线程中执行这个方法。Odb28资讯网——每日最新资讯28at.com

通过AOP的方式实现Async注解的异步执行,Spring框架可以在方法调用时动态地创建代理对象来实现异步执行,而不需要在业务代码中显式地创建新线程。Odb28资讯网——每日最新资讯28at.com

总的来说,Async注解的实现是通过AOP机制来实现的,具体的切面类是AsyncAnnotationAdvisor,它利用AsyncExecutionInterceptor来处理被Async注解标识的方法的调用,实现方法的异步执行。Odb28资讯网——每日最新资讯28at.com

三、Async注解底层异步线程池原理探究

Odb28资讯网——每日最新资讯28at.com

获取Async注解线程池主流程解析

Odb28资讯网——每日最新资讯28at.com

进入到Spring源码Async注解AOP切面实现部分,我们重点剖析异步调用实现中线程池是怎么处理的。下图是org.springframework.aop.interceptor.AsyncExecutionInterceptor#invoke方法的实现,可以看出是调用determineAsyncExecutor方法获取异步线程池。Odb28资讯网——每日最新资讯28at.com

AsyncExecutionInterceptor#invokeAsyncExecutionInterceptor#invokeOdb28资讯网——每日最新资讯28at.com

下图是determineAsyncExecutor方法实现:Odb28资讯网——每日最新资讯28at.com

图片图片Odb28资讯网——每日最新资讯28at.com

图片图片Odb28资讯网——每日最新资讯28at.com

左图为AsyncExecutionInterceptor#determineAsyncExecutor,右图为AsyncExecutionAspectSupport#getExecutorQualifierOdb28资讯网——每日最新资讯28at.com

Odb28资讯网——每日最新资讯28at.com

从代码实现中可以看到determineAsyncExecutor获取线程池的大致流程:Odb28资讯网——每日最新资讯28at.com

determineAsyncExecutor获取线程池流程determineAsyncExecutor获取线程池流程Odb28资讯网——每日最新资讯28at.com

如果在使用Async注解时指定了自定义线程池比较好理解,如果使用Async注解时没有指定自定义线程池,Spring是怎么处理默认线程池呢?继续深入源码看看Spring提供的默认线程池的实现。Odb28资讯网——每日最新资讯28at.com

Odb28资讯网——每日最新资讯28at.com

Spring是怎么为Async注解提供默认线程池的

Odb28资讯网——每日最新资讯28at.com

Async注解默认线程池有下面两个方法实现:   Odb28资讯网——每日最新资讯28at.com

  • org.springframework.aop.interceptor.AsyncExecutionInterceptor#getDefaultExecutor
  • org.springframework.aop.interceptor.AsyncExecutionAspectSupport#getDefaultExecutor

AsyncExecutionInterceptor#getDefaultExecutorAsyncExecutionInterceptor#getDefaultExecutorOdb28资讯网——每日最新资讯28at.com

可以看出AsyncExecutionInterceptor#getDefaultExecutor方法比较简单:先尝试调用父类AsyncExecutionAspectSupport#getDefaultExecutor方法获取线程池,如果父类方法获取不到线程池再用创建SimpleAsyncTaskExecutor对象作为Async的线程池返回。Odb28资讯网——每日最新资讯28at.com

AsyncExecutionAspectSupport#getDefaultExecutorAsyncExecutionAspectSupport#getDefaultExecutorOdb28资讯网——每日最新资讯28at.com

再来看父类AsyncExecutionAspectSupport#getDefaultExecutor方法的实现,可以看到Spring根据类型从Spring容器中获取TaskExecutor类的实例,先记住这个关键点。Odb28资讯网——每日最新资讯28at.com

我们知道,Spring根据类型获取实例时,如果spring容器中有且只有一个指定类型的实例对象,会直接返回,否则的话,会抛出NoUniqueBeanDefinitionException异常或者NoSuchBeanDefinitionException异常。Odb28资讯网——每日最新资讯28at.com

但是,对于Executor类型,Spring容器却“网开一面”,有一个特殊处理:当从Spring容器中获取Executor实例对象时,如果满足@ConditionalOnMissingBean(Executor.class)条件,Spring容器会自动装载一个ThreadPoolTaskExecutor实例对象,而ThreadPoolTaskExecutor是TaskExecutor的实现类。Odb28资讯网——每日最新资讯28at.com

图片图片Odb28资讯网——每日最新资讯28at.com

图片图片Odb28资讯网——每日最新资讯28at.com

左图为TaskExecutionAutoConfiguration,右图为TaskExecutionPropertiesOdb28资讯网——每日最新资讯28at.com

从TaskExecutionProperties和TaskExecutionAutoConfiguration两个配置类我们看到Spring自动装载的ThreadPoolTaskExecutor线程池对象的参数:核心线程数=8;最大线程数=Integer.MAX_VALUE;队列大小=Integer.MAX_VALUE。Odb28资讯网——每日最新资讯28at.com

四、总结

现在Async注解线程池源码已经看的差不多了,下面这张图是Spring处理Async异步线程池的流程:Odb28资讯网——每日最新资讯28at.com

Odb28资讯网——每日最新资讯28at.com

Odb28资讯网——每日最新资讯28at.com

Async异步线程池获取流程Async异步线程池获取流程Odb28资讯网——每日最新资讯28at.com

归纳一下:如果在使用Async注解时没有指定自定义的线程池会出现以下几种情况:Odb28资讯网——每日最新资讯28at.com

  • 当Spring容器中有且仅有一个TaskExecutor实例时,Spring会用这个线程池来处理Async注解的异步任务,这可能会踩坑,如果这个TaskExecutor实例是第三方jar引入的,可能会出现很诡异的问题。
  • Spring创建一个核心线程数=8、最大线程数=Integer.MAX_VALUE、队列大小=Integer.MAX_VALUE的线程池来处理Async注解的异步任务,这时候也可能会踩坑,由于线程池参数设置不合理,核心线程数=8,队列大小过大,如果有大批量并发任务,可能会出现OOM。
  • Spring创建SimpleAsyncTaskExecutor实例来处理Async注解的异步任务,SimpleAsyncTaskExecutor不是一个好的线程池实现类,SimpleAsyncTaskExecutor根据需要在当前线程或者新线程中执行异步任务。如果当前线程已经有空闲线程可用,任务将在当前线程中执行,否则将创建一个新线程来执行任务。由于这个线程池没有线程管理的能力,每次提交任务都实时创建新城,所以如果任务量大,会导致性能下降。

本文链接:http://www.28at.com/showinfo-26-93708-0.html浅析Spring中Async注解底层异步线程池原理

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: React 19 全览,新特性进行一次深度的体验学习

下一篇: 618 入手三星 Galaxy Tab S9 系列 解锁购物「新」方式

标签:
  • 热门焦点
  • 6月iOS设备性能榜:M2稳居榜首 A系列只能等一手3nm来救

    没有新品发布,自然iOS设备性能榜的上榜设备就没有什么更替,仅仅只有跑分变化而产生的排名变动,毕竟苹果新品的发布节奏就是这样的,一年下来也就几个移动端新品,不会像安卓厂商,一
  • 《英雄联盟》夏季赛总决赛今日开打!JDG对阵LNG首发名单来了 Knight:准备三连冠

    8月5日消息,今日17:00,《英雄联盟》2023LPL夏季赛总决赛将正式开打,由JDG对阵LNG。对两支队伍来说,这场比赛不仅要争夺夏季赛冠军,更要决定谁才是LPL赛区一
  • 把LangChain跑起来的三个方法

    使用LangChain开发LLM应用时,需要机器进行GLM部署,好多同学第一步就被劝退了,那么如何绕过这个步骤先学习LLM模型的应用,对Langchain进行快速上手?本片讲解3个把LangChain跑起来
  • 多线程开发带来的问题与解决方法

    使用多线程主要会带来以下几个问题:(一)线程安全问题  线程安全问题指的是在某一线程从开始访问到结束访问某一数据期间,该数据被其他的线程所修改,那么对于当前线程而言,该线程
  • 虚拟键盘 API 的妙用

    你是否在遇到过这样的问题:移动设备上有一个固定元素,当激活虚拟键盘时,该元素被隐藏在了键盘下方?多年来,这一直是 Web 上的默认行为,在本文中,我们将探讨这个问题、为什么会发生
  • 本地生活这块肥肉,拼多多也想吃一口

    出品/壹览商业 作者/李彦编辑/木鱼拼多多也看上本地生活这块蛋糕了。近期,拼多多在App首页“充值中心”入口上线了本机生活界面。壹览商业发现,该界面目前主要
  • 东方甄选单飞:有些鸟注定是关不住的

    文/彭宽鸿编辑/罗卿东方甄选创始人俞敏洪带队的“7天甘肃行”直播活动已在近日顺利收官。成立后一年多时间里,东方甄选要脱离抖音自立门户的传闻不绝于耳,“7
  • 超级标准版旗舰!iQOO 11S全球首发iQOO超算独显芯片

    上半年已接近尾声,截至目前各大品牌旗下的顶级旗舰都已悉数亮相,而下半年即将推出的顶级旗舰已经成为了数码圈爆料的主流,其中就包括全新的iQOO 11S系
  • SN570 NVMe SSD固态硬盘 价格与性能兼具

    SN570 NVMe SSD固态硬盘是西部数据发布的最新一代WD Blue系列的固态硬盘,不仅闪存技术更为精进,性能也得到了进一步的跃升。WD Blue SN570 NVMe SSD的包装外
Top