工做快2年的小白,若有错误,恳请你们批评指点,这也是开始写博客的一个初衷,可以在分享互动、知识梳理中进步。
以前工做的项目使用activiti5进行企业流程系统开发,如今这份工做也开始须要流程开发,了解到activiti6扔掉了原来的pvm,直接针对bpmn进行处理,性能有了必定的提高;也有看到flowable6,虽然说是activiti原班人马出来的,可是实在太新了、用的人也少,小白不敢轻易下手,因而选择了activiti6。说了好多废话。。正题正题java
最为安全可靠,不修改Activiti自身执行和流程定义对象,可是对于中国式流程的功能需求(驳回、回退等),常常是要求在没有连线的状况下完成跳转,灵活性不够。spring
Activiti5中实现,文中第二种
Activiti6中实现
这种方法能够实现动态跳转,不须要修改Activiti自身执行,可是会动态修改系统中的流程定义缓存对象。理论上这会出现一个多线程下,全局变量不安全的问题。单个Activiti流程引擎中,流程定义缓存对象是被全部线程共用的,当一个应用服务器同时收到两个不一样流程实例、同个流程定义、同个环节的任务提交请求。a要求驳回,因此该线程动态修改了流程定义;与此同时,b要求正常流转,可是执行过程当中,依据的流程定义已被修改,可能致使b也走向了驳回。缓存
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实现自由跳转的介绍。后面会再继续介绍服务器
即处理人小王完成环节A的任务(id=6000)后,流程走到下一环节B生成任务(id=6004),任务(id=6004)处理人小张审核不经过执行驳回,流程流转回环节A,环节A从新生成一条id=6000的待处理人为小王的任务。多线程
包括字体配置、自定义全局事件监听、流程定义自动部署开关配置性能