记一次学习Activiti7&SpringBoot

pom依赖
主要引入activiti依赖于jdbc和mysql依赖

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.4.5</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.thz</groupId>
    <artifactId>demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>demo</name>
    <description>Demo project for Spring Boot</description>
    <properties>
        <slf4j.version>1.6.6</slf4j.version>
        <log4j.version>1.2.12</log4j.version>
        <java.version>1.8</java.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.29</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-jdbc</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <dependency>
            <groupId>org.activiti</groupId>
            <artifactId>activiti-spring-boot-starter</artifactId>
            <version>7.0.0.Beta2</version>
        </dependency>

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

application.yml配置

spring:
  datasource:
    url: jdbc:mysql://localhost:3306/activiti?serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8
    username: root
    password: 123456
    driver-class-name: com.mysql.jdbc.Driver
  activiti:
    #1.flase:默认值。activiti在启动时,对比数据库表中保存的版本,如果没有表或者版本不匹配,将抛出异常
    #2.true: activiti会对数据库中所有表进行更新操作。如果表不存在,则自动创建
    #3.create_drop: 在activiti启动时创建表,在关闭时删除表(必须手动关闭引擎,才能删除表)
    #4.drop-create: 在activiti启动时删除原来的旧表,然后在创建新表(不需要手动关闭引擎)
    database-schema-update: true
    #检测历史表是否存在 activiti7默认没有开启数据库历史记录 启动数据库历史记录
    db-history-used: true
    #记录历史等级 可配置的历史级别有none, activity, audit, full #none:不保存任何的历史数据,因此,在流程执行过程中,这是最高效的。
    #activity:级别高于none,保存流程实例与流程行为,其他数据不保存。
    #audit:除activity级别会保存的数据外,还会保存全部的流程任务及其属性。audit为history的默认值。
    #full:保存历史数据的*别,除了会保存audit级别的数据外,还会保存其他全部流程相关的细节数据,包括一些 流程参数等。
    history-level: full
    #校验流程文件,默认校验resources下的processes文件夹里的流程文件
    check-process-definitions: false

准备流程文件 mytest.bpmn
记一次学习Activiti7&SpringBoot

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:activiti="http://activiti.org/bpmn" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:omgdi="http://www.omg.org/spec/DD/20100524/DI" typeLanguage="http://www.w3.org/2001/XMLSchema" expressionLanguage="http://www.w3.org/1999/XPath" targetNamespace="http://www.activiti.org/test">
  <process id="myTest" isClosed="false" isExecutable="true" processType="None">
    <startEvent id="Event_1fb6jx2" name="StartEvent"/>
    <userTask activiti:assignee="zhangsan" activiti:exclusive="true"  id="Activity_0vizqus"  name="创建出差申请单"/>
    <userTask activiti:exclusive="true"  id="Activity_0dm0jl0" activiti:assignee="jack" name="总经理审批"/>
    <endEvent id="Event_1ug08ii" name="EndEvent"/>
    <sequenceFlow id="Flow_13wzyw1" sourceRef="Event_1fb6jx2" targetRef="Activity_0vizqus"/>
    <sequenceFlow id="Flow_09pjrbx" sourceRef="Activity_0vizqus" targetRef="Activity_1aomhta"/>
    <sequenceFlow id="Flow_164iiwb" sourceRef="Activity_0dm0jl0" targetRef="Event_1ug08ii"/>
    <serviceTask id="Activity_1aomhta" name="部门经理审批" activiti:expression="${applyActivitiServiceImpl.finalAudit(execution)}"/>
    <sequenceFlow id="Flow_1p1kzwk" sourceRef="Activity_1aomhta" targetRef="Activity_0dm0jl0"/>
  </process>
  <bpmndi:BPMNDiagram documentation="background=#3C3F41;count=1;horizontalcount=1;orientation=0;width=842.4;height=1195.2;imageableWidth=832.4;imageableHeight=1185.2;imageableX=5.0;imageableY=5.0" id="Diagram-_1" name="New Diagram">
    <bpmndi:BPMNPlane bpmnElement="myTest">
      <bpmndi:BPMNShape bpmnElement="Event_1fb6jx2" id="Shape-Event_1fb6jx2">
        <omgdc:Bounds height="32.0" width="32.0" x="184.0" y="84.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="32.0" width="32.0" x="0.0" y="0.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="Activity_0vizqus" id="Shape-Activity_0vizqus">
        <omgdc:Bounds height="55.0" width="85.0" x="159.0" y="149.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="55.0" width="85.0" x="0.0" y="0.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="Activity_0dm0jl0" id="Shape-Activity_0dm0jl0">
        <omgdc:Bounds height="55.0" width="85.0" x="166.0" y="362.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="55.0" width="85.0" x="0.0" y="0.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="Event_1ug08ii" id="Shape-Event_1ug08ii">
        <omgdc:Bounds height="32.0" width="32.0" x="193.0" y="454.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="32.0" width="32.0" x="0.0" y="0.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="Activity_1aomhta" id="Shape-Activity_1aomhta">
        <omgdc:Bounds height="80.0" width="100.0" x="159.0" y="240.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="80.0" width="100.0" x="0.0" y="0.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNEdge bpmnElement="Flow_164iiwb" id="BPMNEdge_Flow_164iiwb" sourceElement="Activity_0dm0jl0" targetElement="Event_1ug08ii">
        <omgdi:waypoint x="209.0" y="417.0"/>
        <omgdi:waypoint x="209.0" y="454.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="-1.0" width="-1.0" x="-1.0" y="-1.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge bpmnElement="Flow_09pjrbx" id="BPMNEdge_Flow_09pjrbx" sourceElement="Activity_0vizqus" targetElement="Activity_1aomhta">
        <omgdi:waypoint x="201.5" y="204.0"/>
        <omgdi:waypoint x="201.5" y="240.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="-1.0" width="-1.0" x="-1.0" y="-1.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge bpmnElement="Flow_13wzyw1" id="BPMNEdge_Flow_13wzyw1" sourceElement="Event_1fb6jx2" targetElement="Activity_0vizqus">
        <omgdi:waypoint x="200.0" y="116.0"/>
        <omgdi:waypoint x="200.0" y="149.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="-1.0" width="-1.0" x="-1.0" y="-1.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge bpmnElement="Flow_1p1kzwk" id="BPMNEdge_Flow_1p1kzwk" sourceElement="Activity_1aomhta" targetElement="Activity_0dm0jl0">
        <omgdi:waypoint x="208.5" y="320.0"/>
        <omgdi:waypoint x="208.5" y="362.0"/>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="-1.0" width="-1.0" x="-1.0" y="-1.0"/>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNEdge>
    </bpmndi:BPMNPlane>
  </bpmndi:BPMNDiagram>
</definitions>

启动类排除Spring Security
因为Activiti7与SpringBoot整合后,默认情况下,集成了SpringSecurity安全框架,这样我们就要去准备SpringSecurity整合进来的相关用户权限配置信息。这样就很麻烦,这里可以先排除掉。

@SpringBootApplication(
        exclude = {
                org.springframework.boot.autoconfigure.security.servlet.SecurityAutoConfiguration.class,
                org.springframework.boot.actuate.autoconfigure.security.servlet.ManagementWebSecurityAutoConfiguration.class
        }
)

controller

@RestController
public class MyController {
    @Autowired
    ApplyActivitiService applyActivitiService;

    @RequestMapping("/submit")
    public String submitApply(){
        return applyActivitiService.submitApply();
    }

    @RequestMapping("/create")
    public String createApply(String assignee){
        return applyActivitiService.createApply(assignee);
    }

    @RequestMapping("/secondAudit")
    public String secondAudit(String assignee){
        return applyActivitiService.secondAudit(assignee);
    }
}

service

public interface ApplyActivitiService {
    public String submitApply();
    public String createApply(String assignee);
    public String secondAudit(String assignee);
}

实现类

@Service
public class ApplyActivitiServiceImpl implements ApplyActivitiService {

    @Autowired
    ProcessEngine processEngine;

    @Autowired
    RepositoryService repositoryService;

    @Autowired
    RuntimeService runtimeService;

    @Autowired
    TaskService taskService;

    @PostConstruct
    public void init(){
        // 3、使用RepositoryService进行部署
        Deployment deployment = repositoryService.createDeployment()
                .addClasspathResource("bpmn/mytest.bpmn") // 添加bpmn资源
//                .addClasspathResource("bpmn/evection.png") // 添加png资源
                .name("出差申请流程")
                .deploy();
        // 4、输出部署信息
        System.out.println("流程部署id:" + deployment.getId());
        System.out.println("流程部署名称:" + deployment.getName());
    }


    @Override
    public String submitApply() {
        // 3、根据流程定义Id启动流程
        Map<String,Object> variables = new HashMap<>();
        variables.put("myid","123456789");
        ProcessInstance processInstance = runtimeService .startProcessInstanceByKey("myTest",variables);
        // 输出内容
        System.out.println("流程定义id:" + processInstance.getProcessDefinitionId());
        System.out.println("流程实例id:" + processInstance.getId());
        System.out.println("当前活动Id:" + processInstance.getActivityId());
        System.out.println("Name:" + processInstance.getName());
        System.out.println("LocalizedName:" + processInstance.getLocalizedName());
        System.out.println("ProcessDefinitionName:" + processInstance.getProcessDefinitionName());
        return "success";
    }

    @Override
    public String createApply(String assignee) {
        // 根据流程key 和 任务负责人 查询任务
        List<Task> list = taskService.createTaskQuery()
                .processDefinitionKey("myTest") //流程Key
                .taskAssignee(assignee)//只查询该任务负责人的任务
//                .taskId("Activity_0vizqus")
                .list();
        for (Task task : list) {
            System.out.println("流程实例id:" + task.getProcessInstanceId());
            System.out.println("任务id:" + task.getId());
            System.out.println("任务负责人:" + task.getAssignee());
            System.out.println("任务名称:" + task.getName());
            String myid = (String) runtimeService.getVariable(task.getExecutionId(),"myid");
            System.out.println("myid:" + myid);
            Map<String,Object> map = new HashMap<>();
            map.put("bb","BB");
            taskService.complete(task.getId(),map);
        }
        return "scuccess";
    }

    @Override
    public String secondAudit(String assignee) {
        List<Task> list = taskService.createTaskQuery()
                .processDefinitionKey("myTest") //流程Key
                .taskAssignee(assignee)//只查询该任务负责人的任务
//                .taskId("Activity_0dm0jl0")
                .list();
        for (Task task : list) {
            String cc = (String) runtimeService.getVariable(task.getExecutionId(),"cc");
            System.out.println("cc:" + cc);
            taskService.complete(task.getId());
        }
        return "scuccess";
    }

    public void finalAudit(DelegateExecution execution) {
        String bb = (String) execution.getVariable("bb");
        System.out.println("bb:" + bb);
        String myid = (String) execution.getVariable("myid");
        System.out.println("myid:" + myid);
        execution.setVariable("cc","987654321");
    }

}

启动测试

依次调用下面接口

localhost:8080/submit
localhost:8080/create?assignee=zhangsan
localhost:8080/secondAudit?assignee=jack

控制台打印

Connected to the target VM, address: '127.0.0.1:0', transport: 'socket'
SLF4J: The requested version 1.7.16 by your slf4j binding is not compatible with [1.6]
SLF4J: See http://www.slf4j.org/codes.html#version_mismatch for further details.

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::                (v2.4.5)

2021-04-27 14:49:09.326  INFO 15484 --- [           main] com.thz.demo.DemoApplication             : Starting DemoApplication using Java 1.8.0_211 on DESKTOP-0B0L3CT with PID 15484 (D:\test_code\activiti\activitiDemo1\target\classes started by Tang in D:\test_code\activiti\activitiDemo1)
2021-04-27 14:49:09.328  INFO 15484 --- [           main] com.thz.demo.DemoApplication             : No active profile set, falling back to default profiles: default
2021-04-27 14:49:10.256  INFO 15484 --- [           main] trationDelegate$BeanPostProcessorChecker : Bean 'org.springframework.security.access.expression.method.DefaultMethodSecurityExpressionHandler@375084c9' of type [org.springframework.security.access.expression.method.DefaultMethodSecurityExpressionHandler] is not eligible for getting processed by all BeanPostProcessors (for example: not eligible for auto-proxying)
2021-04-27 14:49:10.259  INFO 15484 --- [           main] trationDelegate$BeanPostProcessorChecker : Bean 'methodSecurityConfig' of type [org.activiti.spring.boot.MethodSecurityConfig$$EnhancerBySpringCGLIB$$91e5f2b] is not eligible for getting processed by all BeanPostProcessors (for example: not eligible for auto-proxying)
2021-04-27 14:49:10.264  INFO 15484 --- [           main] trationDelegate$BeanPostProcessorChecker : Bean 'methodSecurityMetadataSource' of type [org.springframework.security.access.method.DelegatingMethodSecurityMetadataSource] is not eligible for getting processed by all BeanPostProcessors (for example: not eligible for auto-proxying)
2021-04-27 14:49:10.412  INFO 15484 --- [           main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat initialized with port(s): 8080 (http)
2021-04-27 14:49:10.417  INFO 15484 --- [           main] o.apache.catalina.core.StandardService   : Starting service [Tomcat]
2021-04-27 14:49:10.418  INFO 15484 --- [           main] org.apache.catalina.core.StandardEngine  : Starting Servlet engine: [Apache Tomcat/9.0.45]
2021-04-27 14:49:10.465  INFO 15484 --- [           main] o.a.c.c.C.[Tomcat].[localhost].[/]       : Initializing Spring embedded WebApplicationContext
2021-04-27 14:49:10.465  INFO 15484 --- [           main] w.s.c.ServletWebServerApplicationContext : Root WebApplicationContext: initialization completed in 1106 ms
2021-04-27 14:49:10.591  INFO 15484 --- [           main] c.t.d.c.DemoApplicationConfiguration     : > Registering new user: jack with the following Authorities[[ROLE_ACTIVITI_USER, GROUP_activitiTeam]]
2021-04-27 14:49:10.658  INFO 15484 --- [           main] c.t.d.c.DemoApplicationConfiguration     : > Registering new user: rose with the following Authorities[[ROLE_ACTIVITI_USER, GROUP_activitiTeam]]
2021-04-27 14:49:10.721  INFO 15484 --- [           main] c.t.d.c.DemoApplicationConfiguration     : > Registering new user: tom with the following Authorities[[ROLE_ACTIVITI_USER, GROUP_activitiTeam]]
2021-04-27 14:49:10.783  INFO 15484 --- [           main] c.t.d.c.DemoApplicationConfiguration     : > Registering new user: other with the following Authorities[[ROLE_ACTIVITI_USER, GROUP_otherTeam]]
2021-04-27 14:49:10.845  INFO 15484 --- [           main] c.t.d.c.DemoApplicationConfiguration     : > Registering new user: system with the following Authorities[[ROLE_ACTIVITI_USER]]
2021-04-27 14:49:10.906  INFO 15484 --- [           main] c.t.d.c.DemoApplicationConfiguration     : > Registering new user: admin with the following Authorities[[ROLE_ACTIVITI_ADMIN]]
2021-04-27 14:49:11.014  INFO 15484 --- [           main] o.s.s.concurrent.ThreadPoolTaskExecutor  : Initializing ExecutorService 'applicationTaskExecutor'
2021-04-27 14:49:11.102  INFO 15484 --- [           main] com.zaxxer.hikari.HikariDataSource       : HikariPool-1 - Starting...
2021-04-27 14:49:11.199  INFO 15484 --- [           main] com.zaxxer.hikari.HikariDataSource       : HikariPool-1 - Start completed.
2021-04-27 14:49:11.611  INFO 15484 --- [           main] aultActiviti5CompatibilityHandlerFactory : Activiti 5 compatibility handler implementation not found or error during instantiation : org.activiti.compatibility.DefaultActiviti5CompatibilityHandler. Activiti 5 backwards compatibility disabled.
2021-04-27 14:49:11.633  INFO 15484 --- [           main] o.a.engine.impl.ProcessEngineImpl        : ProcessEngine default created
2021-04-27 14:49:11.758  INFO 15484 --- [           main] o.a.e.impl.bpmn.deployer.BpmnDeployer    : Process deployed: {id: myTest:2:ac5d7800-a724-11eb-9fa0-70b5e84c1e7c, key: myTest}
流程部署id:ac4fbc5e-a724-11eb-9fa0-70b5e84c1e7c
流程部署名称:出差申请流程
2021-04-27 14:49:12.234  INFO 15484 --- [           main] o.s.b.a.e.web.EndpointLinksResolver      : Exposing 2 endpoint(s) beneath base path '/actuator'
2021-04-27 14:49:12.296  INFO 15484 --- [           main] o.s.b.w.embedded.tomcat.TomcatWebServer  : Tomcat started on port(s): 8080 (http) with context path ''
2021-04-27 14:49:12.307  INFO 15484 --- [           main] com.thz.demo.DemoApplication             : Started DemoApplication in 3.27 seconds (JVM running for 3.685)
2021-04-27 14:49:12.454  INFO 15484 --- [1)-192.168.1.87] o.a.c.c.C.[Tomcat].[localhost].[/]       : Initializing Spring DispatcherServlet 'dispatcherServlet'
2021-04-27 14:49:12.454  INFO 15484 --- [1)-192.168.1.87] o.s.web.servlet.DispatcherServlet        : Initializing Servlet 'dispatcherServlet'
2021-04-27 14:49:12.455  INFO 15484 --- [1)-192.168.1.87] o.s.web.servlet.DispatcherServlet        : Completed initialization in 1 ms
流程定义id:myTest:2:ac5d7800-a724-11eb-9fa0-70b5e84c1e7c
流程实例id:b664e9a1-a724-11eb-9fa0-70b5e84c1e7c
当前活动Id:null
Name:null
LocalizedName:null
ProcessDefinitionName:null
流程实例id:b664e9a1-a724-11eb-9fa0-70b5e84c1e7c
任务id:b6669757-a724-11eb-9fa0-70b5e84c1e7c
任务负责人:zhangsan
任务名称:创建出差申请单
myid:123456789
bb:BB
myid:123456789
cc:987654321

分析

使用@PostConstruct注解在项目启动时自动部署流程
依次执行的三个接口分别执行了启动流程(个人理解为和类 的实例化差不多)、创建出差申请单、总经理审批。因为创建出差申请单和总经理审批为UserTask,需要调用接口执行任务使流程继续向下流动。部门经理审批为ServiceTask,在创建出差申请单后自动完成任务。如果debug启动项目,并在finalAudit方法中打上断点,就会发现在执行完localhost:8080/create?assignee=zhangsan接口后代码并没有执行完,而是进入了finalAudit中的断点。这表明在完成创建申请单任务后,activiti自动去执行下一个ServiceTask了。

关于流程变量,即上面代码中的myid、bb、cc。这些流程变量的作用域为整个流程实例,在设置之后的所有任务都可以使用。通常结合UEL表达式用来控制流程的走向等。
获取某个执行点的变量
RuntimeService
Map<String, Object> getVariables(String executionId);
DelegateExecution
Map<String, Object> getVariables();
Object getVariable(String variableName);
设置某个执行点的变量
RuntimeService
void setVariable(String executionId, String variableName, Object value);
void setVariables(String executionId, Map<String, ? extends Object> variables);
DelegateExecution
void setVariable(String variableName, Object value);
void setVariables(Map<String, ? extends Object> variables);

上一篇:【Luogu U41492】树上数颜色——树上启发式合并(dsu on tree)


下一篇:dsu on tree详解