flowable任务监听器和java-service依赖注入问题

前言

1. Flowable中Java服务任务的依赖注入方法,比如使用Spring的@Autowired或构造函数注入,并确保服务任务类由Spring管理。

2. 流程引擎配置中启用依赖注入和表达式解析的相关设置。

3. 异步任务执行器的配置,确保其与Spring上下文集成,处理依赖注入的Bean。

4. 常见问题排查,如检查Bean的作用域、是否正确使用delegateExpression等。

5. 可能的示例代码或配置片段,帮助用户理解如何正确配置。

另外,可能需要提醒用户注意流程定义文件中的服务任务配置是否正确,比如是否使用了expression或delegateExpression来引用Spring Bean,或者是否直接指定了Java类导致Spring的依赖注入不生效。

最后,如果有必要,可以比较一下在非Spring环境下如何手动处理依赖注入,但大部分情况下用户可能在使用Spring,所以重点放在Spring集成方面。

一、Java服务任务依赖注入

场景 :在实现JavaDelegate的类中注入Spring管理的Bean(如Service、Repository)。

正确配置方法:

  • 步骤1:将服务类注册为Spring Bean
复制代码
@Component("myServiceTask") // 使用名称便于流程中引用
public class MyServiceTask implements JavaDelegate {
    
    private final MyService myService;

    @Autowired // 构造器注入推荐
    public MyServiceTask(MyService myService) {
        this.myService = myService;
    }

    @Override
    public void execute(DelegateExecution execution) {
        myService.businessMethod();
    }
}
  • 步骤2:在流程定义中引用Bean

    使用delegateExpression指向Spring Bean的名称:

复制代码
<serviceTask id="serviceTask1" name="My Task" flowable:delegateExpression="${myServiceTask}" />

常见问题解决:

  • Bean注入为null

    • 确保类被Spring扫描(位于@ComponentScan路径下)。

    • 检查流程定义中是否错误使用class属性而非delegateExpression

    • 确认流程引擎配置启用了Spring集成。

  • 表达式解析失败

    FlowableConfig中启用表达式解析和Spring注入:

复制代码
@Bean
public SpringProcessEngineConfiguration processEngineConfiguration(DataSource dataSource, PlatformTransactionManager transactionManager) {
    SpringProcessEngineConfiguration config = new SpringProcessEngineConfiguration();
    config.setDataSource(dataSource);
    config.setTransactionManager(transactionManager);
    config.setDatabaseSchemaUpdate(Flowable.DB_SCHEMA_UPDATE_TRUE);
    config.setAsyncExecutorEnabled(true); // 启用异步执行器
    config.setExpressionManager(new SpringExpressionManager(applicationContext, null)); // 启用Spring表达式
    return config;
}

二. 异步任务执行器配置

当服务任务标记为异步(flowable:async="true")时,任务由异步执行器处理。

配置异步执行器:

复制代码
@Bean
public AsyncExecutor asyncExecutor(DataSource dataSource) {
    DefaultAsyncExecutor asyncExecutor = new DefaultAsyncExecutor();
    asyncExecutor.setDataSource(dataSource);
    asyncExecutor.setCorePoolSize(5); // 核心线程数
    asyncExecutor.setMaxPoolSize(10); // 最大线程数
    asyncExecutor.setQueueSize(100); // 任务队列大小
    return asyncExecutor;
}

processEngineConfiguration中启用:

复制代码
config.setAsyncExecutor(asyncExecutor);
config.setAsyncExecutorEnabled(true);

异步任务依赖注入要点:

  • Bean作用域 :异步任务可能在新线程中运行,确保注入的Bean是线程安全的或使用@Scope(proxyMode = ScopedProxyMode.TARGET_CLASS)

  • 事务管理 :若异步任务涉及数据库操作,需配置@Transactional并确保事务传播正确。

三、通过ApplicationContextAware接口的方式获取ApplicationContext对象实例

复制代码
可能的错误情况包括:没有在流程引擎配置中设置asyncExecutorEnabled为true,或者在服务任务中没有正确使用表达式导致注入失败。另外,Bean的作用域问题也可能导致依赖注入失败,

例如,如果Bean的作用域是原型(prototype),但在注入时可能需要不同的处理方式。以下是我通过ApplicationContextAware接口的方式获取ApplicationContext对象实例,再通过applicationContext.getBean("myService")

方法获取对应的bean

代码示例:

复制代码
@Component
public class MyListener implements TaskListener, ApplicationContextAware {

    private static  ApplicationContext applicationContext;

    @Override
    public void setApplicationContext(ApplicationContext arg0) throws BeansException {
        applicationContext = arg0;
    }

    @Override
    public void notify(DelegateTask delegateTask) {

        String processInsId = delegateTask.getProcessInstanceId();
        MyService myService = (MyService) applicationContext.getBean("myService");

        // TODO 执行service方法
        
        System.out.println("==========执行监听器======");
    }

}

四. 常见问题排查

  • 错误:无法解析表达式${myServiceTask}

    • 检查Bean名称是否匹配。

    • 确认流程引擎配置中设置了SpringExpressionManager

  • 异步任务不执行

    • 检查asyncExecutor是否启动:调用asyncExecutor.start()

    • 查看日志中是否有任务提交异常。

  • 事务不生效

    • 确保异步执行器配置了事务管理器:asyncExecutor.setTransactionManager(transactionManager);

五. Spring Boot集成

若使用flowable-spring-boot-starter,简化配置如下:

(1) application.yml

复制代码
flowable:
  async-executor-enabled: true
  database-schema-update: true
  async-executor:
    core-pool-size: 5
    max-pool-size: 10
    queue-size: 100

(2) 服务任务类

复制代码
@Component
public class EmailServiceTask implements JavaDelegate {

    private final EmailService emailService;

    public EmailServiceTask(EmailService emailService) {
        this.emailService = emailService;
    }

    @Override
    public void execute(DelegateExecution execution) {
        String recipient = (String) execution.getVariable("email");
        emailService.send(recipient, "流程通知", "您的任务已处理完成。");
    }
}

(3) 流程定义XML

复制代码
<serviceTask id="sendEmail" flowable:delegateExpression="${emailServiceTask}" />

六.总结

  • 依赖注入 :确保服务任务类为Spring Bean,流程中使用delegateExpression引用。

  • 异步执行 :配置AsyncExecutor并启用,注意线程安全和事务。

  • ApplicationContextAware接口的方式获取ApplicationContext对象实例

  • Spring集成 :正确配置SpringProcessEngineConfiguration以支持表达式和Bean解析。

相关推荐
long31610 小时前
java 策略模式 demo
java·开发语言·后端·spring·设计模式
柏油11 小时前
Spring @Cacheable 解读
redis·后端·spring
柏油12 小时前
Spring @TransactionalEventListener 解读
spring boot·后端·spring
白露与泡影14 小时前
Spring容器初始化源码解析
java·python·spring
孤狼程序员20 小时前
【Spring Cloud 微服务】1.Hystrix断路器
java·spring boot·spring·微服务
optimistic_chen1 天前
【Java EE进阶 --- SpringBoot】初识Spring(创建SpringBoot项目)
spring boot·后端·spring·java-ee·tomcat·mvc·idea
期待のcode1 天前
Maven
java·spring·maven·mybatis
郑州吴彦祖7722 天前
Spring MVC快速入门
spring·springmvc
Warren982 天前
MySQL,Redis重点面试题
java·数据库·spring boot·redis·mysql·spring·蓝桥杯
卓码软件测评2 天前
网站测评-利用缓存机制实现XSS的分步测试方法
java·后端·安全·spring·可用性测试·软件需求