activiti学习笔记(九)RuntimeService

RuntimeService 流程运行控制服务

  • 启动流程及对流程数据的控制
  • 流程实例(ProcessInstance)与执行流(Execution)的查询
  • 触发流程操作,接收消息和信号

RuntimeService启动流程及变量管理

  • 启动流程的常用方法(id,key,message)
  • 启动流程可选参数(businessKey,variables,tenantId)
  • 变量(variables)的设置和获取

示例

RuntimeService runtimeService = processEngine.getRuntimeService();

// 根据流程定义id启动流程实例
ProcessInstance processInstance = runtimeService.startProcessInstanceById(processDefinition.getId());

// 根据流程定义key启动流程实例
ProcessInstance processInstance = runtimeService.startProcessInstanceByKey("second_approve");

// 启动流程实例时带上参数
Map<String,Object> variables = new HashMap<String, Object>();
variables.put("key1","value1");
ProcessInstance processInstance = runtimeService.startProcessInstanceByKey("second_approve",variables);
        
// 根据ProcessInstanceBuilder启动流程实例
ProcessInstanceBuilder processInstanceBuilder = runtimeService.createProcessInstanceBuilder();
processInstanceBuilder.businessKey("second_approve")
                .processDefinitionKey("")
                .variables(variables)
                .start();

// 流程启动后获取变量
Map<String,Object> variables1 = runtimeService.getVariables(processInstance.getId());

// 设置变量
runtimeService.setVariable(processInstance.getId(),"key3","value3");

// 查询流程实例对象
runtimeService.createProcessInstanceQuery()
                .processInstanceId(processInstance.getId())
                .singleResult();

// 执行对象查询
List<Execution> executionList = runtimeService.createExecutionQuery()
                .listPage(0,100);
  1. 从示例中可以发现有多种方法可以启动流程,而通过key启动的流程实例每次都是根据最新版本的流程定义启动的,比使用流程定义id启动更加方便。
  2. 流程实例和流程定义和执行对象都一样,都有一个名称后有Query的对象作为查询使用,如果要获取单个结果则调用Query对象的singleResult方法,如果要获取列表数据则调用listPage方法。

流程实例与执行对象

  • 流程实例ProcessInstance表示一次工作流业务的数据实体
  • 执行流Execution表示流程实例中的具体工作路径
  • ProcessInstance类继承Execution类

流程触发

  • 使用trigger触发ReceiveTask节点
  • 触发信号捕获事件signalEventReceived
  • 触发消息捕获事件 messageEventReceived

流程触发示例

首先编写流程文件如下

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:activiti="http://activiti.org/bpmn" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:omgdi="http://www.omg.org/spec/DD/20100524/DI" xmlns:tns="http://www.activiti.org/test" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" expressionLanguage="http://www.w3.org/1999/XPath" id="m1536107421286" name="" targetNamespace="http://www.activiti.org/test" typeLanguage="http://www.w3.org/2001/XMLSchema">
  <process id="second_approve" isClosed="false" isExecutable="true" name="二级审批" processType="None">
    <startEvent id="startEvent" name="开始"/>
    <sequenceFlow id="flow1" sourceRef="startEvent" targetRef="someTask"/>
    <receiveTask id="someTask"></receiveTask>
    <sequenceFlow id="flow2" sourceRef="someTask" targetRef="endEvent"/>
    <endEvent id="endEvent" name="取消"/>
  </process>
</definitions>

流程为开始节点->receiveTask节点->结束节点。
后台实现如下:

ProcessInstance processInstance = runtimeService.startProcessInstanceByKey("second_approve", variables);
Execution execution =runtimeService.createExecutionQuery()
                .activityId("someTask")
                .singleResult();
logger.info("execution={}",execution);
runtimeService.trigger(execution.getId());
execution = runtimeService.createExecutionQuery()
                .activityId("someTask")
                .singleResult();
logger.info("execution={}",execution);

当执行trigger方法后,就执行完receiveTask节点,所以执行前execution对象有值,执行后execution对象就为空了。

事件

事件用来表明流程的生命周期中发生了什么事。 事件总是画成一个圆圈。 在BPMN 2.0 中,事件有两大分类:捕获(catching) 或 触发(throwing) 事件。

  • 捕获(Catching):当流程执行到事件, 它会等待被触发。触发的类型是由内部图表或 XML 中的类型声明来决定的。 捕获事件与触发事件在显示方面是根据内部图表是否被填充来区分的(白色的)。属于捕获的事件有
  • 触发(Throwing):当流程执行到事件, 会触发一个事件。触发的类型是由内部图表或 XML 中的类型声明来决定的。 触发事件与捕获事件在显示方面是根据内部图表是否被填充来区分的(被填充为黑色)。

Event Definitions 事件定义

事件定义决定了事件的语义。如果没有事件定义,这个事件就不做什么特别的事情。 没有设置事件定义的开始事件不会在启动流程时做任何事情。如果给开始事件添加了一个事件定义(比如定时器事件定义)我们就声明了开始流程的事件 “类型 ” (这时定时器事件监听器会在某个时间被触发)。比如错误事件定义(Error Event Definitions)由指定错误触发的。另外还有Signal Event Definitions即信号事件定义。

信号事件

信号事件会引用一个已命名的信号。信号全局范围的事件(广播语义)。 会发送给所有激活的处理器。
信号事件定义使用 signalEventDefinition 元素。 signalRef 属性会引用 definitions 根节点里定义的 signal 子元素。
示例,首先看下流程定义文件xml:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:activiti="http://activiti.org/bpmn" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:omgdi="http://www.omg.org/spec/DD/20100524/DI" xmlns:tns="http://www.activiti.org/test" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" expressionLanguage="http://www.w3.org/1999/XPath" id="m1536107421286" name="" targetNamespace="http://www.activiti.org/test" typeLanguage="http://www.w3.org/2001/XMLSchema">
  <signal id="signalStart" name="my-signal"></signal>
  <process id="second_approve" isClosed="false" isExecutable="true" name="二级审批" processType="None">
    <startEvent id="startEvent" name="开始"/>
    <intermediateCatchEvent id="signal-received">
      <signalEventDefinition signalRef="signalStart"/>
    </intermediateCatchEvent>
    <sequenceFlow id="flow1" sourceRef="startEvent" targetRef="signal-received"/>
    <sequenceFlow id="flow2" sourceRef="signal-received" targetRef="endEvent"/>
    <endEvent id="endEvent" name="取消"/>
  </process>
</definitions>

示例中事件用中间捕获事件封装:

    <intermediateCatchEvent id="myIntermediateCatchEvent" >
          <XXXEventDefinition/>
    </intermediateCatchEvent>

流程文件中定义了一个信号,ID为signalStart.流程启动后就处于等待状态,等待信号事件触发,触发后再继续往下走。
后台代码如下:

Execution execution = runtimeService.createExecutionQuery()
                .signalEventSubscriptionName("my-signal")
                .singleResult();
        logger.info("execution={}",execution);
        runtimeService.signalEventReceived("my-signal");
        execution = runtimeService.createExecutionQuery()
                .signalEventSubscriptionName("my-signal")
                .singleResult();

        logger.info("execution={}",execution);

当执行runtimeService.signalEventReceived方法后就会触发信号,所以执行前后,execution出现由有到无的过程。

消息事件

流程文件定义如下:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:activiti="http://activiti.org/bpmn" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:omgdi="http://www.omg.org/spec/DD/20100524/DI" xmlns:tns="http://www.activiti.org/test" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" expressionLanguage="http://www.w3.org/1999/XPath" id="m1536107421286" name="" targetNamespace="http://www.activiti.org/test" typeLanguage="http://www.w3.org/2001/XMLSchema">
  <message id="messageStart" name="my-message"></message>
  <process id="second_approve" isClosed="false" isExecutable="true" name="二级审批" processType="None">
    <startEvent id="startEvent" name="开始"/>
    <intermediateCatchEvent id="message-received">
      <messageEventDefinition messageRef="messageStart"/>
    </intermediateCatchEvent>
    <sequenceFlow id="flow1" sourceRef="startEvent" targetRef="message-received"/>
    <sequenceFlow id="flow2" sourceRef="message-received" targetRef="endEvent"/>
    <endEvent id="endEvent" name="取消"/>
  </process>
</definitions>

可以看到大致结构和信号事件差不多。
后台代码如下:

Execution execution = runtimeService.createExecutionQuery()
                .messageEventSubscriptionName("my-message")
                .singleResult();
        logger.info("execution={}",execution);
        runtimeService.messageEventReceived("my-message",execution.getId());
        execution = runtimeService.createExecutionQuery()
                .signalEventSubscriptionName("my-message")
                .singleResult();
        logger.info("execution={}",execution);

可以看到信号事件和消息事件写法差不多,区别在于触发事件的时候,除了要传消息的名称外,还要传执行对象id,即不是将事件广播给所有节点,而是指定给某个执行对象。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,649评论 18 139
  •   JavaScript 与 HTML 之间的交互是通过事件实现的。   事件,就是文档或浏览器窗口中发生的一些特...
    霜天晓阅读 3,490评论 1 11
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,047评论 25 707
  • 有朋友写了一篇关于家庭与婚姻的文章,文中说道,30岁以前的婚姻是1+1=1,40岁以前的婚姻是1+1=3,而40岁...
    梅花映雪阅读 453评论 6 8
  • 自大学毕业从事教育工作以来,这是我第三次过教师节。可是,我为什么说这是“第一次”呢,因为龙美给了我太多的第...
    sun艳艳阅读 872评论 0 2