记一次activiti监听器无法实例化的错误解决过程(SpringBoot Listener)

本文记录了一次在SpringBoot项目中使用Activiti监听器遇到的问题,当监听器尝试通过@Autowired注入Bean时,出现实例化错误。经过一系列原因分析、资料查询和尝试,最终确定是由于Listener生命周期由servlet容器管理,而非Spring容器,导致@Autowired注解失效。解决方案是通过手动从Spring容器中获取Bean,从而解决了问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

慢来比较快,虚心学技术

 思路永远比答案更有价值

问题场景

SpringBoot项目中,activiti某个流程节点需要配置监听器,当前节点任务提交完成之后触发方法更改远程表单的状态,具体流程监听设计如下:

feignClient代码:MyService.java

@FeignClient(value = "xxxxx")
public interface MyService {

   /**
    * 修改状态
    * @param inId
    * @return
    */
   @PutMapping("/myTest/updateState")
   Boolean updateState(@RequestParam("id") Integer id);

}

监听器代码:MyTaskListener.java,此处使用Lombok @AllArgConstructor注解通过全参构造器注入MyService

@AllArgsConstructor
public class MyTaskListener implements TaskListener {
    
    private MyService myService;

   /**
    * 任务监听修改表单状态
    * @param delegateTask 当前执行提交的任务
    */
   @Override
   public void notify(DelegateTask delegateTask) {
       。。。。。//获取到表单id
       myService.updateState(id);
   }
}

任务提交完成触发监听器的报错信息:couldn't instantiate class com.xxx.activiti.listener.MyTaskListener

2019-04-16 19:01:45.592 ERROR [xxx-activiti,b4057cc762e15e24,b4057cc762e15e24,true] 224 --- [  XNIO-1 task-1] o.a.e.impl.interceptor.CommandContext    : Error while closing command context

org.activiti.engine.ActivitiException: Exception while invoking TaskListener: couldn't instantiate class com.xxx.activiti.listener.MyTaskListener
	at org.activiti.engine.impl.persistence.entity.TaskEntity.fireEvent(TaskEntity.java:742)
	at org.activiti.engine.impl.persistence.entity.TaskEntity.complete(TaskEntity.java:184)
	at org.activiti.engine.impl.cmd.CompleteTaskCmd.execute(CompleteTaskCmd.java:52)
	at org.activiti.engine.impl.cmd.CompleteTaskCmd.execute(CompleteTaskCmd.java:24)
	at org.activiti.engine.impl.cmd.NeedsActiveTaskCmd.execute(NeedsActiveTaskCmd.java:59)
	at org.activiti.engine.impl.interceptor.CommandInvoker.execute(CommandInvoker.java:24)
	at org.activiti.engine.impl.interceptor.CommandContextInterceptor.execute(CommandCon
### Spring Boot 3 中集成 Activiti 5.22 的教程与示例 Spring Boot 是一个流行的 Java 开源框架,用于简化应用程序开发中的复杂配置。Activiti 则是一个轻量级的工作流和业务流程管理 (BPM) 平台。尽管官方支持可能有所滞后,但在实际项目中仍然可以通过手动调整依赖关系来实现两者的兼容。 #### 1. Maven 依赖配置 为了在 Spring Boot 3 中集成 Activiti 5.22,需要正确引入所需的依赖项。以下是典型的 `pom.xml` 文件片段: ```xml <dependencies> <!-- Spring Boot Starter --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!-- Activiti Core --> <dependency> <groupId>org.activiti</groupId> <artifactId>activiti-spring-boot-starter-basic</artifactId> <version>5.22.0</version> </dependency> <!-- MySQL Connector --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <scope>runtime</scope> </dependency> <!-- HikariCP Connection Pool --> <dependency> <groupId>com.zaxxer</groupId> <artifactId>HikariCP</artifactId> </dependency> </dependencies> ``` 需要注意的是,在 Spring Boot 3 中,默认使用的 JDK 版本较高(通常为 JDK 17 或以上),因此需确认 Activiti 是否完全兼容此环境[^3]。 --- #### 2. 数据库初始化脚本 Activiti 使用数据库存储其运行时数据以及历史录。可以执行以下 SQL 脚本来完成表结构的初始化: ```sql -- 初始化 Activiti 表结构 CREATE DATABASE activiti_db CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci; USE activiti_db; -- 执行建表语句 source path/to/activiti/db/create/mysql/activiti.mysql.create.engine.sql; source path/to/activiti/db/create/mysql/activiti.mysql.create.history.sql; source path/to/activiti/db/create/mysql/activiti.mysql.create.identity.sql; ``` 上述路径可以根据下载的 Activiti 发行版具体位置进行替换。 --- #### 3. 配置文件 (`application.yml`) 通过 YAML 文件定义连接池和其他必要属性: ```yaml spring: datasource: url: jdbc:mysql://localhost:3306/activiti_db?useUnicode=true&characterEncoding=utf8&serverTimezone=UTC username: root password: root driver-class-name: com.mysql.cj.jdbc.Driver jpa: database-platform: org.hibernate.dialect.MySQLDialect hibernate: ddl-auto: none activiti: database-schema-update: true async-executor-enabled: false mail-server-enable: false ``` 此处的关键在于启用自动更新功能并禁用异步执行器以减少不必要的开销[^4]。 --- #### 4. 流程定义与部署 创建一个新的 BPMN 文件作为流程模板,并将其放置于资源目录下(如 `/src/main/resources/processes/leave-process.bpmn20.xml`)。随后可通过如下方式加载该流程: ```java @Configuration public class ActivitiConfig { @Bean public ProcessEngine processEngine() { return ProcessEngines.getDefaultProcessEngine(); } @PostConstruct public void initProcesses() { ProcessEngine engine = processEngine(); RepositoryService repositoryService = engine.getRepositoryService(); // 部署流程定义 repositoryService.createDeployment() .addClasspathResource("processes/leave-process.bpmn20.xml") .deploy(); } } ``` --- #### 5. 自定义监听器 如果希望扩展默认行为,则可编写自定义的任务或事件监听器。例如: ```java @Component public class CustomTaskListener implements TaskListener { @Override public void notify(DelegateTask delegateTask) { System.out.println("触发任务:" + delegateTask.getName()); // 动态分配负责人 if ("审批".equals(delegateTask.getName())) { delegateTask.setAssignee("李四"); } } } ``` 接着将此类绑定到特定节点上即可生效[^4]。 --- #### 6. 启动测试 最后验证整个系统的连通性和基本逻辑是否正常运作。假设存在名为 “请假申请” 的流程实例,那么可以通过 RESTful API 接口发起请求模拟操作过程。 ```bash curl --location 'http://localhost:8080/start-process' \ --header 'Content-Type: application/json' \ --data '{"businessKey":"KEY_001","variables":{"age":29}}' ``` 根据设定好的条件表达式 `${age<=30}` ,不同年龄段会进入相应分支处理阶段[^3]。 --- ###
评论 20
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值