Activiti回退与跳转节点

本文档介绍了如何在Activiti6.0的审批流程中实现当审批不通过时,回退到上一个节点或用户指定的节点的功能,包括查询历史节点、修改流程方向并完成任务撤销操作。

Activiti6.0中审批不通过时回退到上一个节点/指定节点

在当前做的这个审批流中,需要增加一个审批不通过时回退节点的功能,因为当前系统需求还没有特别明确这部分,所以我就先写了以下两种回退情况:

1.回退到上一个节点

 /**
     * 撤回到上一个节点
     * @param processInstanceId
     * @param nowUserId
     */
    @Override
    @Transactional
    public void revoke(String processInstanceId, String nowUserId) {
        //获取待执行的任务节点
        Task task = taskService.createTaskQuery().processInstanceId(processInstanceId).singleResult();
        if(task == null){
           // throw new Exception("sorry,the process is not started or has finished, cannot be withdrawn");
            System.out.println("sorry,the process is not started or has finished, cannot be withdrawn");
        }
        //通过processInstanceId查询历史节点
        List<HistoricTaskInstance> htiList = historyService.createHistoricTaskInstanceQuery()
                .processInstanceId(processInstanceId)
                .orderByTaskCreateTime()
                .asc()
                .list();
        String myTaskId = null;
        HistoricTaskInstance myTask = null;
        //找到当前运行的节点
        HistoricProcessInstance processInstance = historyService.createHistoricProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();
        ProcInstance procInstance = procInstanceMapper.selectById(processInstance.getBusinessKey()).get(0);
        SysUser startUser = sysUserService.selectUserById(procInstance.getUserId());
        List<SysUser> currUsers = findCurrUsers(task, startUser);
        System.out.println("----------- =-= -------currUser-----------------------"+currUsers.get(0).getUserId());
        System.out.println("------ =-= ------nowUserId--------------------------"+nowUserId);
        for (HistoricTaskInstance hti : htiList) {
            //判断一下当前的用户是否为当前任务的审批负责人 , 感觉后面通过权限赋予也可以不加这个判断
            //if (currUsers.get(0).getUserId().equals(nowUserId)&& hti.getId().equals(task.getId())) {
            if ( currUsers.get(0).getUserId().toString().equals(nowUserId)&&hti.getId().equals(task.getId())) {
                myTaskId = hti.getId();  //当前任务id
                myTask = hti;
                break;
            }
        }
        if (null == myTaskId) {
            //throw new Exception("该任务非当前用户提交,无法撤回");
            System.out.println("该任务非当前用户提交,无法撤回~~!!");
        }
        String processDefinitionId = myTask.getProcessDefinitionId();
        //获取流程模型
        BpmnModel bpmnModel = repositoryService.getBpmnModel(processDefinitionId);
        String myActivityId = null;
        //查询已经完成的流程节点,查询到上一条已完成的节点,则跳出循环
        List<HistoricActivityInstance> haiList = historyService.createHistoricActivityInstanceQuery()
                .executionId(myTask.getExecutionId())
                .finished()
                .orderByHistoricActivityInstanceStartTime()
                .desc()
                .list();
        System.out.println("---------------the latest finished---------------"+haiList.get(0));
        myActivityId = haiList.get(0).getActivityId();
//        for (HistoricActivityInstance hai : haiList) {
//            if (myTaskId.equals(hai.getTaskId())) {
//                myActivityId = hai.getActivityId();
//                break;
//            }
//        }
        //最近一个已完成节点的 FlowNode
        FlowNode myFlowNode = (FlowNode) bpmnModel.getMainProcess().getFlowElement(myActivityId);
        //原本的活动方向
        Execution execution = runtimeService.createExecutionQuery().executionId(task.getExecutionId()).singleResult();
        String activityId = execution.getActivityId();
        FlowNode flowNode = (FlowNode) bpmnModel.getMainProcess().getFlowElement(activityId);
        //记录原活动方向
        List<SequenceFlow> oriSequenceFlows = new ArrayList<SequenceFlow>();
        oriSequenceFlows.addAll(flowNode.getOutgoingFlows());
        //清理活动方向
        flowNode.getOutgoingFlows().clear();
        //建立新方向
        List<SequenceFlow> newSequenceFlowList = new ArrayList<SequenceFlow>();
        SequenceFlow newSequenceFlow = new SequenceFlow();
        newSequenceFlow.setId("newSequenceFlowId"+" WITHDRAW: "+nowtime());
        //新方向的源头---当前节点
        newSequenceFlow.setSourceFlowElement(flowNode);
        //新方向的目标---上一个已完成节点
        newSequenceFlow.setTargetFlowElement(myFlowNode);
        newSequenceFlowList.add(newSequenceFlow);
        flowNode.setOutgoingFlows(newSequenceFlowList);

        Authentication.setAuthenticatedUserId(nowUserId);
        taskService.addComment(task.getId(), task.getProcessInstanceId(), "撤回");
        //完成任务
        taskService.complete(task.getId());
        //恢复原方向
        flowNode.setOutgoingFlows(oriSequenceFlows);
        System.out.println("------------------withdraw successfully!!----------------------------");
        logger.info("退回成功!");
    }

2.回退到当前审批用户指定的节点
先查询一下当前流程实例的历史Activity表:

 /**
     * 展示历史Activity表
     * @param proInstanceId
     * @return
     */
    @Override
    @Transactional
    public List hisActInst(String proInstanceId){
        HistoricProcessInstance instance = historyService.createHistoricProcessInstanceQuery().processInstanceBusinessKey(proInstanceId).singleResult();
        String processInstanceId = instance.getId();
        //获取待执行的任务节点
        Task task = taskService.createTaskQuery().processInstanceId(processInstanceId).singleResult();
        if(task == null){
            // throw new Exception("sorry,the process is not started or has finished, cannot be withdrawn");
            System.out.println("sorry,the process is not started or has finished, cannot be withdrawn");
        }
        //通过processInstanceId查询历史节点
        List<HistoricTaskInstance> htiList = historyService.createHistoricTaskInstanceQuery()
                .processInstanceId(processInstanceId)
                .orderByTaskCreateTime()
                .asc()
                .list();
        String myTaskId = null;
        HistoricTaskInstance myTask = null;
        //找到当前运行的节点
        HistoricProcessInstance processInstance = historyService.createHistoricProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();
        ProcInstance procInstance = procInstanceMapper.selectById(processInstance.getBusinessKey()).get(0);
        SysUser startUser = sysUserService.selectUserById(procInstance.getUserId());
        List<SysUser> currUsers = findCurrUsers(task, startUser);
        for (HistoricTaskInstance hti : htiList) {
            //判断一下当前的用户是否为当前任务的审批负责人 , 感觉后面通过权限赋予也可以不加这个判断
            //if (currUsers.get(0).getUserId().equals(nowUserId)&& hti.getId().equals(task.getId())) {
            if ( hti.getId().equals(task.getId())) {
                myTaskId = hti.getId();  //当前任务id
                myTask = hti;
                break;
            }
        }
        if (null == myTaskId) {
            //throw new Exception("该任务非当前用户提交,无法撤回");
            System.out.println("该任务非当前用户提交,无法撤回~~!!");
        }
        List<HistoricActivityInstance> haiList = historyService.createHistoricActivityInstanceQuery()
                .executionId(myTask.getExecutionId())
                .finished()
                .orderByHistoricActivityInstanceStartTime()
                .desc()
                .list();
        return haiList;
    }


在根据获取到的目标回退节点的ActivityId进行回退操作:

/**
     * 回退到指定节点处
     * @param processInstanceId
     * @param nowUserId
     * @param tarActivityId
     * @return
     */
    @Override
    @Transactional
    public void rollBackToSpec(String processInstanceId, String nowUserId, String tarActivityId) {
        //获取待执行的任务节点
        Task task = taskService.createTaskQuery().processInstanceId(processInstanceId).singleResult();
        if(task == null){
            // throw new Exception("sorry,the process is not started or has finished, cannot be withdrawn");
            System.out.println("sorry,the process is not started or has finished, cannot be withdrawn");
        }
        //通过processInstanceId查询历史节点
        List<HistoricTaskInstance> htiList = historyService.createHistoricTaskInstanceQuery()
                .processInstanceId(processInstanceId)
                .orderByTaskCreateTime()
                .asc()
                .list();
        String myTaskId = null;
        HistoricTaskInstance myTask = null;
        //找到当前运行的节点
        HistoricProcessInstance processInstance = historyService.createHistoricProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();
        ProcInstance procInstance = procInstanceMapper.selectById(processInstance.getBusinessKey()).get(0);
        SysUser startUser = sysUserService.selectUserById(procInstance.getUserId());
        List<SysUser> currUsers = findCurrUsers(task, startUser);
        System.out.println("----------- =-= -------currUser-----------------------"+currUsers.get(0).getUserId());
        System.out.println("------ =-= ------nowUserId--------------------------"+nowUserId);
        for (HistoricTaskInstance hti : htiList) {
            //判断一下当前的用户是否为当前任务的审批负责人 , 感觉后面通过权限赋予也可以不加这个判断
            //if (currUsers.get(0).getUserId().equals(nowUserId)&& hti.getId().equals(task.getId())) {
            if ( currUsers.get(0).getUserId().toString().equals(nowUserId)&&hti.getId().equals(task.getId())) {
                myTaskId = hti.getId();  //当前任务id
                myTask = hti;
                break;
            }
        }
        if (null == myTaskId) {
            //throw new Exception("该任务非当前用户提交,无法撤回");
            System.out.println("该任务非当前用户提交,无法撤回~~!!");
        }
        String processDefinitionId = myTask.getProcessDefinitionId();
        //获取流程模型
        BpmnModel bpmnModel = repositoryService.getBpmnModel(processDefinitionId);
//        String myActivityId = null;
        //查询已经完成的流程节点,查询到上一条已完成的节点,则跳出循环
//        List<HistoricActivityInstance> haiList = historyService.createHistoricActivityInstanceQuery()
//                .executionId(myTask.getExecutionId())
//                .finished()
//                .orderByHistoricActivityInstanceStartTime()
//                .desc()
//                .list();
//        System.out.println("---------------the latest finished---------------"+haiList.get(0));
//        myActivityId = haiList.get(0).getActivityId();
        //想要回退到的节点位置
        System.out.println(tarActivityId);
        FlowNode myFlowNode = (FlowNode) bpmnModel.getMainProcess().getFlowElement(tarActivityId);
        //原本的活动方向
        Execution execution = runtimeService.createExecutionQuery().executionId(task.getExecutionId()).singleResult();
        String activityId = execution.getActivityId();
        FlowNode flowNode = (FlowNode) bpmnModel.getMainProcess().getFlowElement(activityId);
        //记录原活动方向
        List<SequenceFlow> oriSequenceFlows = new ArrayList<SequenceFlow>();
        oriSequenceFlows.addAll(flowNode.getOutgoingFlows());
        //清理活动方向
        flowNode.getOutgoingFlows().clear();
        //建立新方向
        List<SequenceFlow> newSequenceFlowList = new ArrayList<SequenceFlow>();
        SequenceFlow newSequenceFlow = new SequenceFlow();
        newSequenceFlow.setId("newSequenceFlowId"+" ROLLBACK: "+nowtime());
        //新方向的源头---当前节点
        newSequenceFlow.setSourceFlowElement(flowNode);
        System.out.println("--------------new flow-----------------"+flowNode);
        //新方向的目标---要回退的节点
        System.out.println("--------------target flow-----------------"+myFlowNode);
        newSequenceFlow.setTargetFlowElement(myFlowNode);
        newSequenceFlowList.add(newSequenceFlow);
        flowNode.setOutgoingFlows(newSequenceFlowList);

        Authentication.setAuthenticatedUserId(nowUserId);
        taskService.addComment(task.getId(), task.getProcessInstanceId(), "回退");
        //完成任务
        System.out.println("========================完成任务====================");
        taskService.complete(task.getId());

        //恢复原方向
        flowNode.setOutgoingFlows(oriSequenceFlows);
        System.out.println("------------------RollBack successfully!!----------------------------");
        logger.info("退回成功!");
        //将退回完成后的当前节点的task返回
//        HistoricActivityInstance historicActivityInstance = historyService.createHistoricActivityInstanceQuery().activityId(tarActivityId).singleResult();
//
//        return historicActivityInstance.getTaskId();
    }


<think>我们正在讨论Activiti工作流引擎中如何实现流程跳转到任意节点的问题。根据引用[3]的内容,Activiti提供了节点跳转的功能,允许从节点跳转到另节点,即使它们之间没有直接的连线。 在Activiti中,流程的跳转通常通过`RuntimeService`的`createProcessInstanceModification`方法来实现。这允许我们在流程实例运行时修改其执行路径。具体步骤如下: 1. 获取当前流程实例的ID。 2. 使用`RuntimeService`创建个流程实例修改对象(`ProcessInstanceModificationBuilder`)。 3. 指定要取消的当前活动节点(如果需要)。 4. 指定要跳转到的目标节点(使用节点ID)。 5. 执行修改。 这里有个简单的代码示例: ```java // 获取RuntimeService RuntimeService runtimeService = ... // 通过流程引擎获取 String processInstanceId = "当前流程实例ID"; String targetNodeId = "目标节点ID"; // 例如:usertask5 // 创建修改 runtimeService.createProcessInstanceModification(processInstanceId) .cancelActivityInstance("当前活动节点实例ID") // 可选,如果当前节点只有个活动实例,可以通过活动实例ID取消 .startBeforeActivity(targetNodeId) // 跳转到目标节点之前,即启动目标节点 .execute(); ``` 或者,如果我们知道当前的任务ID,我们可以先获取当前的任务,然后获取当前的活动节点实例,再进行跳转。 另种方法是使用`ActivityInstance`来获取当前的活动节点实例信息,然后进行跳转。 注意:跳转操作需要谨慎,因为它会改变流程的正常路径,可能会影响流程数据的完整性。因此,在执行跳转前,需要确保目标节点是合法的,并且跳转后流程的状态仍然是致的。 另外,Activiti还提供了通过`Execution`对象进行跳转的方式,例如: ```java List<Execution> executions = runtimeService.createExecutionQuery() .processInstanceId(processInstanceId) .activityId("当前节点ID") // 例如:usertask3 .list(); for (Execution execution : executions) { runtimeService.createChangeActivityStateBuilder() .processInstanceId(processInstanceId) .moveExecutionTo(execution.getId(), targetNodeId) .changeState(); } ``` 以上代码中,我们首先查询到当前正在执行的执行流(execution),然后使用`createChangeActivityStateBuilder`来改变执行流的状态,将其移动到目标节点。 需要注意的是,这些操作都需要在流程实例处于运行状态时进行,并且跳转的目标节点必须在流程定义中存在。 在实际应用中,我们可能需要根据业务需求选择合适的跳转方式,并处理跳转后可能产生的数据变更和后续流程的走向。 §§相关问题§§ 1. Activiti中流程跳转有哪些注意事项? 2. 如何回退到上节点? 3. Activiti中流程跳转后,流程变量如何处理?</think>在 Activiti 工作流引擎中实现流程跳转到任意节点的功能,可以通过以下两种核心方法实现: ### 方法:使用 `RuntimeService.createProcessInstanceModification`(推荐) ```java String processInstanceId = "当前流程实例ID"; String targetNodeId = "目标节点ID"; // 如 usertask5 runtimeService.createProcessInstanceModification(processInstanceId) .cancelAllForActivity("当前活动节点ID") // 取消当前节点 .startBeforeActivity(targetNodeId) // 启动目标节点 .execute(); ``` **说明**: 1. `cancelAllForActivity()` 终止当前节点所有实例 2. `startBeforeActivity()` 在目标节点前创建新执行流 3. 适用于 Activiti 5.17+ 版本[^3] ### 方法二:使用 `CommandContext`(底层API) ```java ProcessEngineConfigurationImpl config = (ProcessEngineConfigurationImpl) ProcessEngines.getDefaultProcessEngine().getProcessEngineConfiguration(); config.getCommandExecutor().execute(commandContext -> { ExecutionEntity execution = ...; // 获取当前执行实例 execution.destroyScope("跳转原因"); // 创建新执行路径 ExecutionEntity newExecution = execution.createExecution(); newExecution.setCurrentFlowElement( processDefinition.findActivityElement(targetNodeId) ); // 触发节点进入事件 newExecution.executeActivityBehavior(); return null; }); ``` **说明**: 1. 直接操作执行实例(`ExecutionEntity`) 2. 需要手动处理流程作用域和事件触发 3. 适用于需要深度定制场景[^2] ### 关键注意事项 1. **节点有效性**:目标节点必须在流程定义中存在 2. **变量处理**:跳转后需手动同步流程变量 ```java runtimeService.setVariables(newExecutionId, variablesMap); ``` 3. **事件触发**:自动节点(如ServiceTask)需手动触发`execute()` 4. **历史记录**:跳转操作会生成`ACT_HI_ACTINST`历史记录 5. **并发控制**:多实例场景需遍历所有执行实例 > 重要提示:流程跳转属于高风险操作,会破坏正常流程路径,建议仅在调试或特殊业务场景使用。生产环境应优先考虑通过`BPMN 2.0`规范的`boundary event`或`sequenceFlow`实现合法流转[^1]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

princeAladdin

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值