19

前言

工作快2年的小白,如有错误,恳请大家批评指点,这也是开始写博客的一个初衷,能够在分享互动、知识梳理中进步。
之前工作的项目使用activiti5进行企业流程系统开发,现在这份工作也开始需要流程开发,了解到activiti6扔掉了原来的pvm,直接针对bpmn进行处理,性能有了一定的提升;也有看到flowable6,虽说是activiti原班人马出来的,但是实在太新了、用的人也少,小白不敢轻易下手,于是选择了activiti6。说了好多废话。。正题正题

Activiti实现跳转的方式有三种:

1.流程图配置连线,正常流转

最为安全可靠,不修改Activiti自身执行和流程定义对象,但是对于中国式流程的功能需求(驳回、回退等),经常是要求在没有连线的情况下完成跳转,灵活性不够。

2.动态修改流程定义环节的连线,然后执行跳转,完成后再恢复流程定义

Activiti5中实现,文中第二种
Activiti6中实现
这种方法可以实现动态跳转,不需要修改Activiti自身执行,但是会动态修改系统中的流程定义缓存对象。理论上这会出现一个多线程下,全局变量不安全的问题。单个Activiti流程引擎中,流程定义缓存对象是被所有线程共用的,当一个应用服务器同时收到两个不同流程实例、同个流程定义、同个环节的任务提交请求。a要求驳回,所以该线程动态修改了流程定义;与此同时,b要求正常流转,但是执行过程中,依据的流程定义已被修改,可能导致b也走向了驳回。

3.直接指定当前流程实例执行所选的环节

Activiti5中实现
Activiti5中实现,文中第一种
这种方法即可以实现动态跳转,又没有动态修改流程定义带来的不安全问题,而activiti6中因为pvm下的包都删了,执行计划的代码也进行了改造,原来的方法就不可用了。没找到相关介绍,自己根据原先的思路,学习Activiti6源码,找到了实现的方法,下面提供代码。

//跳转方法
public void jump(String taskId){
        //当前任务
        Task currentTask = taskService.createTaskQuery().taskId(taskId).singleResult();
        //获取流程定义
        Process process = repositoryService.getBpmnModel(currentTask.getProcessDefinitionId()).getMainProcess();
        //获取目标节点定义
        FlowNode targetNode = (FlowNode)process.getFlowElement("startTask");
        //删除当前运行任务
        String executionEntityId = managementService.executeCommand(new DeleteTaskCmd(currentTask.getId()));
        //流程执行到来源节点
        managementService.executeCommand(new SetFLowNodeAndGoCmd(targetNode, executionEntityId));
    }
------------------
//删除当前运行时任务命令,并返回当前任务的执行对象id
//这里继承了NeedsActiveTaskCmd,主要时很多跳转业务场景下,要求不能时挂起任务。可以直接继承Command即可
public class DeleteTaskCmd extends NeedsActiveTaskCmd<String> {
    public DeleteTaskCmd(String taskId){
        super(taskId);
    }
    public String execute(CommandContext commandContext, TaskEntity currentTask){
        //获取所需服务
        TaskEntityManagerImpl taskEntityManager = (TaskEntityManagerImpl)commandContext.getTaskEntityManager();
        //获取当前任务的来源任务及来源节点信息
        ExecutionEntity executionEntity = currentTask.getExecution();
        //删除当前任务,来源任务
        taskEntityManager.deleteTask(currentTask, "jumpReason", false, false);
        return executionEntity.getId();
    }
    public String getSuspendedTaskException() {
        return "挂起的任务不能跳转";
    }
}
------------------
//根据提供节点和执行对象id,进行跳转命令
public class SetFLowNodeAndGoCmd implements Command<Void> {
    private FlowNode flowElement;
    private String executionId;
    public SetFLowNodeAndGoCmd(FlowNode flowElement,String executionId){
        this.flowElement = flowElement;
        this.executionId = executionId;
    }

    public Void execute(CommandContext commandContext){
        //获取目标节点的来源连线
        List<SequenceFlow> flows = flowElement.getIncomingFlows();
        if(flows==null || flows.size()<1){
            throw new ActivitiException("回退错误,目标节点没有来源连线");
        }
        //随便选一条连线来执行,时当前执行计划为,从连线流转到目标节点,实现跳转
        ExecutionEntity executionEntity = commandContext.getExecutionEntityManager().findById(executionId);
        executionEntity.setCurrentFlowElement(flows.get(0));
        commandContext.getAgenda().planTakeOutgoingSequenceFlowsOperation(executionEntity, true);
        return null;
    }
}

最后

以上就是对Activiti6实现自由跳转的介绍。后面会再继续介绍

1.以上述自由跳转为基础实现不改变原先任务id的驳回

即处理人小王完成环节A的任务(id=6000)后,流程走到下一环节B生成任务(id=6004),任务(id=6004)处理人小张审核不通过执行驳回,流程流转回环节A,环节A重新生成一条id=6000的待处理人为小王的任务。

2.java类方式进行Activiti6配置、spring集成

包括字体配置、自定义全局事件监听、流程定义自动部署开关配置


坐死等吃
190 声望183 粉丝