以前写过一篇文章activiti根据当前节点获取下一个节点信息,当时的需求只是需要获取下一个节点而已,并没有要求获得什么类型的节点,所以下一个节点可能是任何节点,在最近的项目中的需求是根据当前的节点获取下一个UseTask节点,为当前审批人选择下一个审批人提示,所以前面的文章所给出的信心就不能够实现当前功能。
所以本文所述为获取下一个任务节点。根据上一篇文章,可以获取到下一个节点的信息,如果我们获取到的节点是Exclusive Gateway,那么再根据这个节点再往下查询直到下一个节点是useTask为止,那么遇到GateWay就需要根据条件对流程进行判断,获取el表达式,再根据条件得到流程走的是哪一条线。

一、测试流程图:


流程图中最重要的是需要设置gateWay的Id值–这个Id值与gatway流出的flow的el表达式的key要对应(不然程序在执行中会出错)。



下面就是具体的执行程序:

/** * 获取下一个用户任务信息  * @param String taskId     任务Id信息  * @return  下一个用户任务用户组信息  * @throws Exception */  public TaskDefinition getNextTaskInfo(String taskId) throws Exception {  ProcessDefinitionEntity processDefinitionEntity = null;  String id = null;  TaskDefinition task = null;  //获取流程实例Id信息   String processInstanceId = taskService.createTaskQuery().taskId(taskId).singleResult().getProcessInstanceId();  //获取流程发布Id信息   String definitionId = runtimeService.createProcessInstanceQuery().processInstanceId(processInstanceId).singleResult().getProcessDefinitionId();  processDefinitionEntity = (ProcessDefinitionEntity) ((RepositoryServiceImpl) repositoryService)  .getDeployedProcessDefinition(definitionId);  ExecutionEntity execution = (ExecutionEntity) runtimeService.createProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();  //当前流程节点Id信息   String activitiId = execution.getActivityId();    //获取流程所有节点信息   List<ActivityImpl> activitiList = processDefinitionEntity.getActivities();   //遍历所有节点信息   for(ActivityImpl activityImpl : activitiList){      id = activityImpl.getId();     if (activitiId.equals(id)) {//获取下一个节点信息   task = nextTaskDefinition(activityImpl, activityImpl.getId(), null, processInstanceId); break;}}  return task;  }  /**  * 下一个任务节点信息,  *  * 如果下一个节点为用户任务则直接返回,  *  * 如果下一个节点为排他网关, 获取排他网关Id信息, 根据排他网关Id信息和execution获取流程实例排他网关Id为key的变量值,  * 根据变量值分别执行排他网关后线路中的el表达式, 并找到el表达式通过的线路后的用户任务* @param ActivityImpl activityImpl     流程节点信息  * @param String activityId             当前流程节点Id信息  * @param String elString               排他网关顺序流线段判断条件* @param String processInstanceId      流程实例Id信息  * @return  */    private TaskDefinition nextTaskDefinition(ActivityImpl activityImpl, String activityId, String elString, String processInstanceId){   PvmActivity ac = null;Object s = null;// 如果遍历节点为用户任务并且节点不是当前节点信息if ("userTask".equals(activityImpl.getProperty("type")) && !activityId.equals(activityImpl.getId())) {// 获取该节点下一个节点信息TaskDefinition taskDefinition = ((UserTaskActivityBehavior) activityImpl.getActivityBehavior()).getTaskDefinition();return taskDefinition;} else {// 获取节点所有流向线路信息List<PvmTransition> outTransitions = activityImpl.getOutgoingTransitions();List<PvmTransition> outTransitionsTemp = null;for (PvmTransition tr : outTransitions) {ac = tr.getDestination(); // 获取线路的终点节点// 如果流向线路为排他网关if ("exclusiveGateway".equals(ac.getProperty("type"))) {outTransitionsTemp = ac.getOutgoingTransitions();// 如果网关路线判断条件为空信息if (StringUtils.isEmpty(elString)) {// 获取流程启动时设置的网关判断条件信息elString = getGatewayCondition(ac.getId(), processInstanceId);}// 如果排他网关只有一条线路信息if (outTransitionsTemp.size() == 1) {return nextTaskDefinition((ActivityImpl) outTransitionsTemp.get(0).getDestination(), activityId,elString, processInstanceId);} else if (outTransitionsTemp.size() > 1) { // 如果排他网关有多条线路信息for (PvmTransition tr1 : outTransitionsTemp) {s = tr1.getProperty("conditionText"); // 获取排他网关线路判断条件信息// 判断el表达式是否成立if (isCondition(ac.getId(), StringUtils.trim(s.toString()), elString)) {return nextTaskDefinition((ActivityImpl) tr1.getDestination(), activityId, elString,processInstanceId);}}}} else if ("userTask".equals(ac.getProperty("type"))) {return ((UserTaskActivityBehavior) ((ActivityImpl) ac).getActivityBehavior()).getTaskDefinition();} else {}}return null;}}  /** * 查询流程启动时设置排他网关判断条件信息  * @param String gatewayId          排他网关Id信息, 流程启动时设置网关路线判断条件key为网关Id信息  * @param String processInstanceId  流程实例Id信息  * @return */  public String getGatewayCondition(String gatewayId, String processInstanceId) {  Execution execution = runtimeService.createExecutionQuery().processInstanceId(processInstanceId).singleResult();Object object= runtimeService.getVariable(execution.getId(), gatewayId);return object==null? "":object.toString();  }  /** * 根据key和value判断el表达式是否通过信息  * @param String key    el表达式key信息  * @param String el     el表达式信息  * @param String value  el表达式传入值信息  * @return */  public boolean isCondition(String key, String el, String value) {  ExpressionFactory factory = new ExpressionFactoryImpl();    SimpleContext context = new SimpleContext();    context.setVariable(key, factory.createValueExpression(value, String.class));    ValueExpression e = factory.createValueExpression(context, el, boolean.class);    return (Boolean) e.getValue(context);  }  

单元测试:

@Testpublic void nextTaskInfo() throws Exception{ActivitiNextTask task = new ActivitiNextTask();String processInstanceId = "387905";TaskDefinition nextTaskGroup = task.getNextTaskInfo(processInstanceId);System.out.println(nextTaskGroup.getKey());}

通过测试能够正确的根据条件获取下一个usetask节点信息。
那么又存在另一种情况,如果流程中存在连续多个GateWay,那么上述的程序获取下一个节点就不正确了,所以需要在上述的程序上在进行修改。

二、流程图二


在费用报销申请后面有两个GateWay,当流程到达费用报销申请时需要判断下一步是审批还是审批(部门经理)对于第二个GateWay需要修改的是:


然后再对上述的代码进行修改:

private TaskService taskService = ProcessEngines.getDefaultProcessEngine().getTaskService() ;private RuntimeService runtimeService = ProcessEngines.getDefaultProcessEngine().getRuntimeService();private RepositoryService repositoryService = ProcessEngines.getDefaultProcessEngine().getRepositoryService();/** * 获取下一个用户任务信息  * @param String taskId     任务Id信息  * @return  下一个用户任务用户组信息  * @throws Exception */  public TaskDefinition getNextTaskInfo(String processInstanceId) throws Exception {  ProcessDefinitionEntity processDefinitionEntity = null;  String id = null;  TaskDefinition task = null;  //获取流程实例Id信息   //String processInstanceId = taskService.createTaskQuery().taskId(taskId).singleResult().getProcessInstanceId();  //获取流程发布Id信息   String definitionId = runtimeService.createProcessInstanceQuery().processInstanceId(processInstanceId).singleResult().getProcessDefinitionId();  processDefinitionEntity = (ProcessDefinitionEntity) ((RepositoryServiceImpl) repositoryService)  .getDeployedProcessDefinition(definitionId);  ExecutionEntity execution = (ExecutionEntity) runtimeService.createProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();  //当前流程节点Id信息   String activitiId = execution.getActivityId();    //获取流程所有节点信息   List<ActivityImpl> activitiList = processDefinitionEntity.getActivities();   //遍历所有节点信息   for(ActivityImpl activityImpl : activitiList){      id = activityImpl.getId();     if (activitiId.equals(id)) {//获取下一个节点信息   task = nextTaskDefinition(activityImpl, activityImpl.getId(), null, processInstanceId); break;}}  return task;  }  /**  * 下一个任务节点信息,  *  * 如果下一个节点为用户任务则直接返回,  *  * 如果下一个节点为排他网关, 获取排他网关Id信息, 根据排他网关Id信息和execution获取流程实例排他网关Id为key的变量值,  * 根据变量值分别执行排他网关后线路中的el表达式, 并找到el表达式通过的线路后的用户任务* @param ActivityImpl activityImpl     流程节点信息  * @param String activityId             当前流程节点Id信息  * @param String elString               排他网关顺序流线段判断条件* @param String processInstanceId      流程实例Id信息  * @return  */    private TaskDefinition nextTaskDefinition(ActivityImpl activityImpl, String activityId, String elString, String processInstanceId){   PvmActivity ac = null;Object s = null;// 如果遍历节点为用户任务并且节点不是当前节点信息if ("userTask".equals(activityImpl.getProperty("type")) && !activityId.equals(activityImpl.getId())) {// 获取该节点下一个节点信息TaskDefinition taskDefinition = ((UserTaskActivityBehavior) activityImpl.getActivityBehavior()).getTaskDefinition();return taskDefinition;} else if("exclusiveGateway".equals(activityImpl.getProperty("type"))){// 当前节点为exclusiveGatewayList<PvmTransition> outTransitions = activityImpl.getOutgoingTransitions();//outTransitionsTemp = ac.getOutgoingTransitions();// 如果网关路线判断条件为空信息
//          if (StringUtils.isEmpty(elString)) {// 获取流程启动时设置的网关判断条件信息elString = getGatewayCondition(activityImpl.getId(), processInstanceId);
//          }// 如果排他网关只有一条线路信息if (outTransitions.size() == 1) {return nextTaskDefinition((ActivityImpl) outTransitions.get(0).getDestination(), activityId,elString, processInstanceId);} else if (outTransitions.size() > 1) { // 如果排他网关有多条线路信息for (PvmTransition tr1 : outTransitions) {s = tr1.getProperty("conditionText"); // 获取排他网关线路判断条件信息// 判断el表达式是否成立if (isCondition(activityImpl.getId(), StringUtils.trim(s.toString()), elString)) {return nextTaskDefinition((ActivityImpl) tr1.getDestination(), activityId, elString,processInstanceId);}}}}else {// 获取节点所有流向线路信息List<PvmTransition> outTransitions = activityImpl.getOutgoingTransitions();List<PvmTransition> outTransitionsTemp = null;for (PvmTransition tr : outTransitions) {ac = tr.getDestination(); // 获取线路的终点节点// 如果流向线路为排他网关if ("exclusiveGateway".equals(ac.getProperty("type"))) {outTransitionsTemp = ac.getOutgoingTransitions();// 如果网关路线判断条件为空信息if (StringUtils.isEmpty(elString)) {// 获取流程启动时设置的网关判断条件信息elString = getGatewayCondition(ac.getId(), processInstanceId);}// 如果排他网关只有一条线路信息if (outTransitionsTemp.size() == 1) {return nextTaskDefinition((ActivityImpl) outTransitionsTemp.get(0).getDestination(), activityId,elString, processInstanceId);} else if (outTransitionsTemp.size() > 1) { // 如果排他网关有多条线路信息for (PvmTransition tr1 : outTransitionsTemp) {s = tr1.getProperty("conditionText"); // 获取排他网关线路判断条件信息// 判断el表达式是否成立if (isCondition(ac.getId(), StringUtils.trim(s.toString()), elString)) {return nextTaskDefinition((ActivityImpl) tr1.getDestination(), activityId, elString,processInstanceId);}}}} else if ("userTask".equals(ac.getProperty("type"))) {return ((UserTaskActivityBehavior) ((ActivityImpl) ac).getActivityBehavior()).getTaskDefinition();} else {}}return null;}return null;}  /** * 查询流程启动时设置排他网关判断条件信息  * @param String gatewayId          排他网关Id信息, 流程启动时设置网关路线判断条件key为网关Id信息  * @param String processInstanceId  流程实例Id信息  * @return */  public String getGatewayCondition(String gatewayId, String processInstanceId) {  Execution execution = runtimeService.createExecutionQuery().processInstanceId(processInstanceId).singleResult();Object object= runtimeService.getVariable(execution.getId(), gatewayId);return object==null? "":object.toString();  }  /** * 根据key和value判断el表达式是否通过信息  * @param String key    el表达式key信息  * @param String el     el表达式信息  * @param String value  el表达式传入值信息  * @return */  public boolean isCondition(String key, String el, String value) {  ExpressionFactory factory = new ExpressionFactoryImpl();    SimpleContext context = new SimpleContext();    context.setVariable(key, factory.createValueExpression(value, String.class));    ValueExpression e = factory.createValueExpression(context, el, boolean.class);    return (Boolean) e.getValue(context);  }  

编写测试用例:

@Testpublic void startProcessInstance(){//流程定义的keyString processDefinitionKey = "exclusiveGateWay";Map<String, Object> param = new HashMap<>();//${creditor=='group'} 添加流程变量,选定流程路线param.put("creditor", "group");param.put("subCreditor", "a");processEngine.getRuntimeService().startProcessInstanceByKey(processDefinitionKey,param);}

按照上述的测试用例流程到达-费用报销申请,下一步应该是审批【部门经理】

@Testpublic void nextTaskInfo() throws Exception{ActivitiNextTask task = new ActivitiNextTask();String processInstanceId = "387905";TaskDefinition nextTaskGroup = task.getNextTaskInfo(processInstanceId);System.out.println(nextTaskGroup.getNameExpression().getExpressionText());}

activiti根据当前节点获取下一个UseTask节点相关推荐

  1. activiti根据当前节点获取下一个节点信息

    在流程中使用监听器判断当前节点是否需要经过,否则跳转到下一个节点,如下图 当提交申请之后,当前提交人为部门负责人,那么部门负责人节点就不需要走了,直接到下一个节点,但是下一个节点是什么并不知道,就可以 ...

  2. 2.<tag-二叉树和层序遍历相关题>-lt.116. 填充每个节点的下一个右侧节点指针 + lt.117. 填充每个节点的下一个右侧节点指针 II 1

    lt.116. 填充每个节点的下一个右侧节点指针 [案例需求] [思路分析一, 迭代法] [代码实现] /* // Definition for a Node. class Node {public ...

  3. leetcode116. 填充每个节点的下一个右侧节点指针

    116. 填充每个节点的下一个右侧节点指针 难度中等128 给定一个完美二叉树,其所有叶子节点都在同一层,每个父节点都有两个子节点.二叉树定义如下: struct Node {int val;Node ...

  4. 116. 填充每个节点的下一个右侧节点指针

    2020-05-28 1.题目描述 填充每个节点的下一个右侧节点指针 2.题解 层次遍历即可 3.代码 /* // Definition for a Node. class Node { public ...

  5. 力扣116. 填充每个节点的下一个右侧节点指针(C++,附思路)

    题目链接在这里: 116. 填充每个节点的下一个右侧节点指针 - 力扣(LeetCode) (leetcode-cn.com)https://leetcode-cn.com/problems/popu ...

  6. 力扣 填充每个节点的下一个右侧节点指针

    填充每个节点的下一个右侧节点指针 题目描述 给定一个完美二叉树,其所有叶子节点都在同一层,每个父节点都有两个子节点.二叉树定义如下: struct Node { int val; Node *left ...

  7. [leetcode-117]填充每个节点的下一个右侧节点指针 II

    (1 AC) 填充每个节点的下一个右侧节点指针 I是完美二叉树.这个是任意二叉树 给定一个二叉树 struct Node {int val;Node *left;Node *right;Node *n ...

  8. 二叉树:填充每个节点的下一个右侧节点指针(java)

    leetcode116:填充每个节点的下一个右侧节点指针 leetcode原题链接: 题目描述 递归解法一 递归方法二(效率更高) 二叉树专题 leetcode原题链接: 116题:填充每个节点的下一 ...

  9. html 获取下一个兄弟节点,js jquery获取当前元素的兄弟级 上一个 下一个元素

    var chils= s.childNodes;  //得到s的全部子节点 var par=s.parentNode;   //得到s的父节点 var ns=s.nextSbiling;   //获得 ...

最新文章

  1. 华为Mate 30系列发布:电影级4摄,麒麟990 5G,没谷歌全家桶,售价高于iPhone11
  2. [WorldWind学习]12.WavingFlags和WavingFlagLayer
  3. Bamboolib -- 十分钟教会业务小姐姐做数据分析
  4. 一个情怀引发的生产事故
  5. 自由职业半年,我又滚回职场了...
  6. html页面送值转换,在静态页面html中跳转传值
  7. free mybatis 不生效_这些Mybatis-Plus的使用小技巧,你应该知道的!
  8. 为什么创业你只为少数人服务就够了?
  9. 504 Gateway Time-out 错误处理记录
  10. 【笔记17】使用 jad 工具把 java 的 class 文件转换为 java 文件;使用 luyten 把 java 的 jar 包转换为 java文件
  11. LightningChart数据可视化工具图形控件教程47-SurfaceGridSeries3D(曲线网格系列3D)
  12. ShardingJdbc SQLFeatureNotSupportedException: isValid
  13. Oracle EBS APP-FND-02901: you don‘t have privilege to any operating unit
  14. gcr.io 国内源 —— 基于 Travis CI + GitHub + DockerHub + Google Cloud 实现自动定时同步 gcr.io 镜像到 DockerHub
  15. codevs 2867 天平系统3
  16. CSS经常会用到的属性
  17. 解决 cannot connect to 192.168.1.136:5555: 由于目标计算机积极拒绝,无法连接。 (10061)
  18. 修改host访问远景论坛
  19. VMware vRealize Automation 8 Install
  20. 多行溢出隐藏显示省略号功能的JS实现

热门文章

  1. Tinux 3.0 内存对齐算法
  2. linux中256错误,YUM安装遭遇: [Errno 256] No more mirrors to try
  3. 尝试自动重定向的次数太多_GoRod:基于DP协议的Web自动化和数据抓取工具
  4. ESD静电二极管,在汽车电子中的应用
  5. matlab朴素贝叶斯手写数字识别_TensorFlow手写数字识别(一)
  6. select使用css居右对齐
  7. GPU云服务器是什么?谷歌宣布将要推出GPU云服务器
  8. PCA人脸识别的python实现
  9. 【翻译自mos文章】job 不能自己主动运行的解决方法
  10. C# - linq查询现有的DataTable