Activity代码实现

简介: 本教程演示如何使用Activiti创建、部署并操作简单工作流。包括流程定义、启动实例、任务查询与完成,结合代码与数据库验证,实现审批流程自动化管理。

1 创建bpmn流程

  1. resources下新建一个文件夹,名称随意,这里我们叫:bpmn
  2. 右键,新创建一个:bpmnFile,名称随意,这里我们叫:hello.bpmn

  1. 拖拽生成流程节点信息,这里我们简单操作,视频操作流程如下:

此处为语雀视频卡片,点击链接查看:bpmn操作視頻.mp4


好了,截止目前我们就生成了一个简单的,单节点审批的工作流。接下来我们就将以此流程作为Demo,开始我们的学习之路。

2 部署流程定义

此代码是模拟我们开发好一套审批流程,后续员工就可以发起这个流程

请确保你已正确配置数据库连接(端口、用户名、密码),数据库处于启动状态,并且在本地创建了对应的数据库:activiti(名称可以随意)。

代码编写

@Test
public void deploymentProcessDefinition(){
    //创建ProcessEngine对象
    ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
    //获取RepositoryService对象
    RepositoryService repositoryService = processEngine.getRepositoryService();
    //进行部署
    Deployment deployment = repositoryService
            .createDeployment()
            .name("测试入门案例")
            .addClasspathResource("bpmn/hello.bpmn")
            .deploy();
    //输出部署的一些信息
    System.out.println("流程部署ID:"+deployment.getId());
    System.out.println("流程部署名称:"+deployment.getName());
}

上述代码唯一需要注意的地方,就是:addClasspathResource 中的路径,如果读者们与我保持不一致,请记得及时修改。

代码测试

运行单测,会发现日志中打印了大量的DDL语句,这是因为Activiti作为一款成熟的工作流引擎框架,他已经有了完善的数据库:

执行完成我们在最后的日志中看到输出结果:

此时我们去刷新数据库,会发现多了25张表,如下(大概表分类我们放在:04-Activiti拓展篇讲解,这里只讲最简单的上手Demo):

此时,最关键的来了:我们去:act_re_procdef中会发现有我们刚才的流程定义:

这就验证了我们的流程定义代码测试成功。相当于我们公司现在有了一个单节点的审批流程,后面就等着用户发起,这个流程就可以使用起来。

3 启动流程实例

此代码模拟的就是某个用户开始提交流程,此时就会开启某个流程,如下

@Test
public void startProcessInstance(){
    //获取与正在执行的流程示例和执行对象相关的Service
    ProcessInstance processInstance = ProcessEngines.getDefaultProcessEngine().getRuntimeService()
            //使用流程定义的key启动实例,key对应bpmn文件中id的属性值,默认按照最新版本流程启动
            .startProcessInstanceByKey("myProcess_1");
    System.out.println("流程实例ID:"+processInstance.getId());
    System.out.println("流程实例定义ID:"+processInstance.getProcessDefinitionId());
}

这里面需要注意的是:流程实例的key,对应表:act_re_procdef.key字段,二者必须保持一致,也就是bpmn文件id属性值。

当我们执行完成,输出打印如下:

此时我们去:act_ru_task 表中可以看到生成一个新的task数据,即当前用户开启了一个新的流程

同时可以查看到流程的多个任务节点信息:act_ru_execution

4 查询当前任务

此代码模拟员工查询目前审批在哪个节点,好催促对应审批人员进行审批

@Test
public void findPersonalTask(){
    //与正在执行的任务相关的Service
    List<Task> list = ProcessEngines.getDefaultProcessEngine().getTaskService()
            .createTaskQuery()  //创建查询任务对象
//          .taskAssignee("UserTask")  //指定个人任务查询,指定办理人
            .list();
    if(list != null && list.size() > 0){
        for(Task task : list){
            System.out.println("任务ID" + task.getId());
            System.out.println("任务名称" + task.getName());
            System.out.println("任务创建时间" + task.getCreateTime());
            System.out.println("任务指定人" + task.getAssignee());
            System.out.println("任务流程实例ID" + task.getProcessInstanceId());
            System.out.println("任务执行ID" + task.getExecutionId());
            System.out.println("任务流程定义ID" + task.getProcessDefinitionId());
        }
    }
}

运行后代码输出如下:

5 完成指定任务

此代码模拟某个用户完成对应的节点,即:审批通过/拒绝,流程就会自动流转至下一个节点

@Test
public void completePersonalTask(){
    // 对应task表的id
    ProcessEngines.getDefaultProcessEngine()
            .getTaskService()
            .complete("2505");
}

需要注意的是,此id对应的是上述:4 查询当前任务 返回任务ID保持一致。

当运行完成后,表:act_ru_task、act_ru_execution这种进行中数据都会清空,进入到:act_hi_**表中。


以上我们就完成了一个简单的工作流引擎代码编写与测试。

6 完整代码汇总

package com.demo;
import org.activiti.engine.ProcessEngine;
import org.activiti.engine.ProcessEngines;
import org.activiti.engine.RepositoryService;
import org.activiti.engine.repository.Deployment;
import org.activiti.engine.runtime.ProcessInstance;
import org.activiti.engine.task.Task;
import org.junit.Test;
import java.util.List;
public class ActivitiTest {
    @Test
    public void deploymentProcessDefinition(){
        //创建ProcessEngine对象
        ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
        //获取RepositoryService对象
        RepositoryService repositoryService = processEngine.getRepositoryService();
        //进行部署
        Deployment deployment = repositoryService
                .createDeployment()
                .name("测试入门案例")
                .addClasspathResource("bpmn/hello.bpmn")
                .deploy();
        //输出部署的一些信息
        System.out.println("流程部署ID:"+deployment.getId());
        System.out.println("流程部署名称:"+deployment.getName());
    }
    @Test
    public void startProcessInstance(){
        //获取与正在执行的流程示例和执行对象相关的Service
        ProcessInstance processInstance = ProcessEngines.getDefaultProcessEngine().getRuntimeService()
                //使用流程定义的key启动实例,key对应bpmn文件中id的属性值,默认按照最新版本流程启动
                .startProcessInstanceByKey("myProcess_1");
        System.out.println("流程实例ID:"+processInstance.getId());
        System.out.println("流程实例定义ID:"+processInstance.getProcessDefinitionId());
    }
    @Test
    public void findPersonalTask(){
        //与正在执行的任务相关的Service
        List<Task> list = ProcessEngines.getDefaultProcessEngine().getTaskService()
                .createTaskQuery()  //创建查询任务对象
//          .taskAssignee("UserTask")  //指定个人任务查询,指定办理人
                .list();
        if(list != null && list.size() > 0){
            for(Task task : list){
                System.out.println("任务ID" + task.getId());
                System.out.println("任务名称" + task.getName());
                System.out.println("任务创建时间" + task.getCreateTime());
                System.out.println("任务指定人" + task.getAssignee());
                System.out.println("任务流程实例ID" + task.getProcessInstanceId());
                System.out.println("任务执行ID" + task.getExecutionId());
                System.out.println("任务流程定义ID" + task.getProcessDefinitionId());
            }
        }
    }
    @Test
    public void completePersonalTask(){
        // 对应task表的id
        ProcessEngines.getDefaultProcessEngine()
                .getTaskService()
                .complete("2505");
    }
    /**
     * 1-2-3-4
     *
     * 2-1-3-4
     * 3-1-2-4
     * 流程编排、流程可视化
     */
}
相关文章
|
11小时前
|
监控 算法 Unix
Thread.sleep(0) 到底有什么
Thread.Sleep用于让线程暂停执行一段时间,不参与CPU竞争。Sleep(1000)并不保证精确唤醒时间,因系统调度受优先级和资源影响;而Sleep(0)则触发系统立即重新分配CPU,给予其他线程执行机会,避免界面假死。理解其原理有助于优化多线程程序性能与响应性。
|
12小时前
|
Java Nacos Maven
Eureka服务注册与发现
本章介绍Eureka注册中心的搭建与使用,完成user-service和order-service服务注册,并实现多实例部署。虽Eureka已被SpringCloud逐步淘汰,但其核心思想仍具参考价值,后续将用Nacos替代深入探讨。
|
12小时前
|
监控 算法 Unix
Thread.sleep(0) 到底有什么用(读完就懂)
Thread.Sleep用于暂停线程执行,Sleep(1000)不保证精确唤醒时间,因系统调度受优先级和竞争影响;Sleep(0)则触发立即重新调度,让出CPU给其他线程,避免界面假死。二者作用显著不同。
|
12小时前
|
人工智能 搜索推荐 自动驾驶
FPGA 的回归:AI 时代,算力终于开始“量体裁衣”了
FPGA 的回归:AI 时代,算力终于开始“量体裁衣”了
9 0
|
12小时前
|
存储 关系型数据库 MySQL
幂等方案专题
本文简要介绍了关系型与非关系型数据库的区别、索引的作用及类型、B+树与B树的差异、MySQL的事务机制、隔离级别、锁机制、性能优化策略等内容,涵盖数据库核心概念与常见面试问题,帮助理解数据库设计与应用中的关键知识点。
|
14小时前
|
SQL 运维 分布式计算
如何做好SQL质量监控
本文介绍如何通过CloudLens for SLS实现SQL质量监控,帮助用户全面掌握SQL使用情况。涵盖健康分、服务指标、运行明细、SQL Pattern分析及优化建议五大维度,助力识别性能瓶颈、优化资源使用,提升日志分析效率与治理能力。
 如何做好SQL质量监控
|
14小时前
|
运维 Devops 开发工具
生产环境缺陷管理
git-poison是一款基于go-git的生产环境缺陷管理工具,实现bug的分布式追溯与自动化卡点,有效避免多分支开发中的漏修、漏发问题,降低协同成本,提升发布安全性。
生产环境缺陷管理
|
11小时前
|
Web App开发 Dubbo 关系型数据库
Soul网关接入与验证
本章节在前两章基础上集成Soul网关,讲解Provider与Consumer应用如何通过添加依赖、注解及配置文件接入Soul,实现服务注册与网关路由,并通过实际请求验证网关调用流程。
Soul网关接入与验证
|
14小时前
|
Java 测试技术 Linux
生产环境发布管理
本文介绍大型团队如何通过自动化部署平台实现多环境(dev/test/pre/prod)高效发布,涵盖各环境职责、基于Jenkins+K8S的CI/CD流程、分支管理与日志排查方案,提升发布效率与系统稳定性。
 生产环境发布管理
|
11小时前
|
Arthas Java 测试技术
基础指令
Arthas 是一款Java诊断工具,支持查看线程信息(dashboard、thread)、反编译类(jad)、观测方法调用(watch)等。可通过命令查看最忙线程、阻塞线程、指定状态线程及方法入参出参,支持采样间隔设置,帮助快速定位性能问题。
基础指令

热门文章

最新文章