activiti获取可回退的节点

时间:2022-01-25 01:57:46

在处理流程回退时,需要获取某个节点当前可以回退到的节点,简单分析下:

1. 只支持回退到userTask。

2. 如果流程流转过某节点时生成了多个任务,从其中某一个任务回退到该节点后,不处理另外的任务。

3. 只能回退到当前节点前已经处理过的节点。

基于这个考虑,获取哪些节点可以回退按如下处理:

1. 从历史任务表查当前节点所在流程实例已经完成过的所有任务,有可能包括当前节点以后的任务(比如当前节点本来就是由后面节点回退的)

2. 判断任务是不是当前节点以前的userTask节点任务,如果是,加入返回列表。

3. 同一个节点只返回一次。

在这种处理前提下,需要获取流程中所有userTask节点的前后关系。我用深度优先的方式对流程的所有节点进行了一个遍历,把所有节点的前后关系保存在一个map里,map的内容为:

key: 前面的节点编号#后面的节点编号

value:从前面节点到后面节点中间要经过几个节点。

如果一个节点前有多个节点,比如 流程是 A-->B-->C-->D,那么 map的数据类似:

{A#B:1,A#C:2,A#D:3,B#C:1,B#D:2,C#D:1}

在这种情况下,所有的可能性都存储了一份,但是在后面判断节点的前后关系时相对比较容易,只要把要判断的节点和当前节点编号拼接在一起,判断 要判断的节点#当前节点 是不是在map里存在就好了。

一般来说,流程的节点数不会太多,这样已经够用了。

由于回退不是在流程上通过划线处理的,所以一般不会出现环状的流程图,因此用深度优先方式遍历也适用,感觉上应该用广度优先的方式遍历更合适一点。

附部分代码:

private static Map<String,Map<String,Object>> PROCESS_NODE_MAP = null;
private Map getFlowNodeRel(String processDefKey){
if (null == PROCESS_NODE_MAP){
PROCESS_NODE_MAP = new HashMap<String, Map<String, Object>>();
}
if(null == PROCESS_NODE_MAP.get(processDefKey)){
ProcessEngine engine = ProcessEngines.getDefaultProcessEngine();
List<Process> processList = engine.getRepositoryService().getBpmnModel(processDefKey).getProcesses();
Process process = null;
if(processList.size()>0){
process = processList.get(0);
Map relationMap = new HashMap<String,Integer>();
setFlowNodeRel(process,relationMap);
PROCESS_NODE_MAP.put(processDefKey, relationMap);
}
}
return PROCESS_NODE_MAP.get(processDefKey);
} //根据process往relationMap里写入UserTask之间的前后关系,从start开始遍历节点,不循环
private void setFlowNodeRel(Process process,Map relationMap){
Iterator<FlowElement> iterator = process.getFlowElements().iterator();
HashMap nodeMap = new HashMap<String,FlowElement>();
setFlowNodeMap(process,nodeMap);
//获取start节点
StartEvent startEvent = (StartEvent) nodeMap.get("startEvent");
//处理节点关系
setFlowNodeRel("","startEvent",nodeMap,relationMap,1);
} //根据节点ID设置关系
private void setFlowNodeRel(String parentKey,String nodeId,Map<String,FlowElement> nodeMap,Map relationMap,int level){
FlowElement element = nodeMap.get(nodeId);
if(element instanceof UserTask){
relationMap.put(nodeId, element);
}
if(!(element instanceof FlowNode)){
return;
} List<SequenceFlow> outFlows = ((FlowNode)element).getOutgoingFlows();
if(outFlows.size() == 0){
return;
} //采用深度遍历
for(int index=0;index < outFlows.size();index++){
SequenceFlow tmpSequence = outFlows.get(index);
String target = tmpSequence.getTargetRef();
String key = nodeId + "#" + target; String loopKey = target + "#" + nodeId; FlowElement tmpElement = nodeMap.get(target);
//只是源和目标都是 用户任务 才写入
if(relationMap.containsKey(loopKey) || relationMap.containsKey(key)){
continue;
}
if(tmpElement instanceof UserTask && element instanceof UserTask){
setRelMapValue(relationMap,parentKey,target, level);
}
//递归找下一个任务的关系
setFlowNodeRel(parentKey+"#"+target,target,nodeMap,relationMap,level+1);
}
} private void setRelMapValue(Map relationMap,String parentKey,String nodeKey,int level){
if(parentKey.length() > 0){
String[] parentArr = parentKey.split("#");
for(String parent:parentArr){
if(StringUtils.isNotEmpty(parent)){
String key = parent + "#" + nodeKey;
String loopKey = nodeKey + "#" + parent;
if(relationMap.containsKey(loopKey) || relationMap.containsKey(key)){
continue;
}else{
relationMap.put(key, level);
}
}
}
}
} //把流程节点设置到map属性里
private void setFlowNodeMap(Process process,Map<String,FlowElement> nodeMap){
Iterator<FlowElement> iterator = process.getFlowElements().iterator(); while(iterator.hasNext()){
FlowElement element = iterator.next();
if(element instanceof StartEvent){
nodeMap.put("startEvent", element);
}else{
nodeMap.put(element.getId(),element);
}
}
}

activiti获取可回退的节点的更多相关文章

  1. activiti获取流程图中的所有信息包括变量名

    今天在看activiti的变量的时候,都是由用户来设置变量,我们在后台写代码通过变量设置或取一些参数的时候,非常的不方便. 比如:设置变量的时候通过页面设置 设置之后,我们如何动态的给这个变量设置参数 ...

  2. 解析某些特殊格式XML文件时,获取不到根节点问题

    还是在语音识别这块.在读取本地的SRGS的XML后,无法获取到根节点<grammar>. 下面是SRGS.XML文件(只给出了根节点) <?xml version="1.0 ...

  3. 获取Json数据某节点的值

    时间匆忙,直接上代码,回家还得做清蒸鱼呢! #region 获取Json字符串某节点的值 /// <summary> /// 获取Json字符串某节点的值 /// </summary ...

  4. I&period;MX6 Linux 自动获取AR1020 event input节点

    /*********************************************************************** * I.MX6 Linux 自动获取AR1020 ev ...

  5. 获取所有树叶子节点 注册添加事件 if &lpar;&dollar;&lpar;node&rpar;&period;tree&lpar;&&num;39&semi;isLeaf&&num;39&semi;&comma; node&period;target&rpar;&rpar; 是否叶子节点

    //获取所有树叶子节点 注册添加事件 if ($(node).tree('isLeaf', node.target)) 是否叶子节点 $(function () { $('.easyui-tree') ...

  6. react&period;js 获取真实的DOM节点

    为了获取真实的dom节点,文本输入框必须有一个 ref 属性,然后 this.refs.[refName] 就会返回这个真实的 DOM 节点. var MyComponent = React.crea ...

  7. vue&lowbar;elementUI&lowbar; tree树形控件 获取选中的父节点ID

    el-tree 的 this.$refs.tree.getCheckedKeys() 只可以获取选中的id 无法获取选中的父节点ID想要获取选中父节点的id;需要如下操作1. 找到工程下的node_m ...

  8. JavaScript的DOM&lowbar;获取和操作层次节点

    一.层次节点的概述 节点的层次结构可以划分为:父节点与子节点.兄弟节点这两种.当我们获取其中一个元素节点的时候,就可以使用层次节点属性来获取它相关层次的节点. 二.childNodes 属性 chil ...

  9. 获取iframe子页面节点,响应浏览器宽高

    获取iframe子页面节点,响应浏览器宽高 html部分代码 <div> <iframe width="100%" height="100%" ...

随机推荐

  1. &lbrack;No000056&rsqb;你无法真正占有一个人,包括你的爱人,先生或太太、小孩,以及你自己&period;&period;&period;&period;

    从一出生,我们的双手就握的紧紧的,好像深知自己会失去什么 很多人迷信多子多孙才是福,老来才有依靠,但太多新闻告诉我们,很多人老了,子孙为了分家产,反而让他生不如死,死了还无法入土为安. 现实也告诉我们 ...

  2. 9&period;配置postfix空客户端

    将本地邮件服务器配置充当为控客户端,已将所有邮件都转发到*服务器以进行发送 1.postconf -e "relayhost=[mail.example.com]" 邮件被路由到 ...

  3. 程序启动原理和UIApplication

    iOS开发UI篇—程序启动原理和UIApplication   一.UIApplication 1.简单介绍 (1)UIApplication对象是应用程序的象征,一个UIApplication对象就 ...

  4. 解决蛋疼的阿里云单CPU使用率的问题。

    工作中涉及到阿里云的应用.在性能测试阶段,压测过程中只要一个CPU未使用满,第二个CPU以至于第三个和第四个CPU完全用不到. 后来和阿里云的同事沟通他们现在用的是单队列的网卡,只能靠RPS/RFS这 ...

  5. Query获取值常用

    Query获取Select选择的Text和Value:语法解释:1. $("#select_id").change(function(){//code...});   //为Sel ...

  6. Python学习之---冒泡,选择,插入排序

    Python学习之---冒泡,选择,插入排序 最近学习了python基础,写一下3大排序练练手: 1 ''' 2 Created on 2013-8-23 3 4 @author: codegeek ...

  7. jsonp实现跨域资源共享原理

  8. Java异常、事件、多线程

    异常     捕捉异常,以便程序继续执行,同时可进行异常处理使程序更加健壮.     Throwble类,派生Exception类和Error类,Exception类供应用程序用,Error类系统保留 ...

  9. 【转】一口气读懂NB-IoT

    在过去的一年多,NB-IoT真的可以说是大红大紫.在通信圈里,除了说5G,就是说物联网.如果说物联网,八成就是在说NB-IoT. 在目前5G还没来的情况下,NB-IoT基本上是独领风骚.风光无限. 各 ...

  10. C&num; 笔记 Func&lt&semi;TResult&gt&semi; 委托、Action&lt&semi;T&gt&semi; 委托

    https://blog.csdn.net/wanglui1990/article/details/79303894 Func<ΤResult> 委托:代理(delegate)一个返回类型 ...