【问题篇】Parameter 0 of method springAsyncExecutor in *.* equired a single bean, but 2 were found

文章讲述了在SpringBoot项目中使用Scheduled定时任务和Activiti工作流时,由于自定义线程池和内置定时任务导致的TaskExecutor多实例问题,通过添加@Primary注解解决了冲突,确保了Activiti正确使用配置的线程池。
摘要由CSDN通过智能技术生成

互相学习交流 

        最近遇到一个小问题,再次记录一下,以便于后面遇到了可以回顾起来。

        背景:在一个springboot项目上,使用了boot自带的定时任务(Scheduled),并且整合了activiti工作流,并且在项目中自己注册了线程池的bean。


项目现状:

  • 1、开启了springboot自带的定时任务

  • 2、在项目中配置了自己的线程池(注意:我此处没有使用的juc包下的ThreadPoolExecutor,而是springboot包装后的ThreadPoolTaskExecutor,后面的代码描述都是针对此情况。)

  • 3、 项目中整合了activiti

        针对这种情况下启动项目,会发生什么呢?

出现情况:

  • 1、情况

  • 2、情况

        Error creating bean with name 'springAsyncExecutor' defined in class path resource [org/activiti/spring/boot/ProcessEngineAutoConfiguration.class]: Unsatisfied dependency expressed through method 'springAsyncExecutor' parameter 0;nested exception is org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'org.springframework.core.task.TaskExecutor' available: expected single matching bean but found 2: taskExecutor,taskScheduler (ps:太长,用文本表示)

        结果显然报错了。

        针对报错:

  1. 截图意思是AbstractProcessEngineAutoConfiguration这类的bean注册时,需要一个TaskExecutor类型的单例bean的参数,但是却发现了两个,不知道使用哪一个。
  2. 文本意思是创建ProcessEngineAutoConfiguration这个bean时发生错误,他的一个参数bean创建也失败了,原因和上面一样,他的参数springAsyncExecutor也需要一个TaskExecutor类型的单例bean的参数,但是却发现了两个,不知道使用哪一个。
  3. 出现上面报错其实就是因为TaskExecutor的bean注入出现了多个导致activiti注册自己的bean报错。

        我们点进这些类看一下。

针对情况1:

        发现就是这个TaskExecutor参数,为什么会出现两个呢,我们再一起看一下。找到TaskSchedulingAutoConfiguration类。

        当我们使用@EnableScheduling注解时,同时会向spring注册internalScheduledAnnotationProcessor这个处理器(已经测试过),当满足这个情况,存在@ConditionalOnBean注解,他就会向spring注册一个ThreadPoolTaskScheduler的bean,而这个bean就是一个TaskExecutor。

        意味着,当开启定时任务后,spring容器中就已经存在一个TaskExecutor类型的bean了。

        再然后我们配置的线程池注册的也同样是一个TaskExecutor类型的bean。

        而activiti的AbstractProcessEngineAutoConfiguration类不知道需要哪一个bean,因此报错,而我们也无法帮助他指定,只得修改我们的配置类。

针对情况2:

        打开ProcessEngineAutoConfiguration类

        可以看到和上面一样的情况,都是无法确定唯一的TaskExecutor类型的bean。

解决方式:

        通过加上@Primary注解可以解决,我们来指定以这个bean为主。

        我们再重启项目打断点试一下。

针对情况1:

        根据此处的参数可以看出AbstractProcessEngineAutoConfiguration使用的正是我们配置类注册进去的bean。

针对情况2:

        可以看到ProcessEngineAutoConfiguration也是使用的我们配置类注册进去的bean。

        结果:项目启动成功。

        我们再实验一下,当我们不自定义线程池时。我们找到这个类TaskExecutionAutoConfiguration

        可以看到他注册了两个bean,一个是预先包装ThreadPoolTaskExecutor的参数bean,一个是生成一个ThreadPoolTaskExecutorbean,启动项目时情况如下:

        可以看到activiti正常情况下使用的正是springboot自动装配产生的ThreadPoolTaskExecutorBean。

        欢迎交流。

  • 25
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小z♂

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值