activiti工作流会签例子

本文详细介绍了一个基于BPMN的请假申请流程实现,包括流程图的绘制、Java代码示例及流程变量的设置。通过多实例并行会签、条件判断等特性,实现了灵活的流程控制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.bpmn文件

<?xml version="1.0" encoding="UTF-8"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" 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" typeLanguage="http://www.w3.org/2001/XMLSchema" expressionLanguage="http://www.w3.org/1999/XPath" targetNamespace="http://www.activiti.org/test">
  <process id="applyHoliday2" name="请假申请" isExecutable="true">
    <startEvent id="startevent1" name="Start"></startEvent>
    <userTask id="usertask1" name="申请"></userTask>
    <sequenceFlow id="flow1" sourceRef="startevent1" targetRef="usertask1"></sequenceFlow>
    <userTask id="usertask2" name="会签" activiti:assignee="${signer}">
      <multiInstanceLoopCharacteristics isSequential="false" activiti:collection="${signList}" activiti:elementVariable="signer">
        <completionCondition>${nrOfCompletedInstances==1}</completionCondition>
      </multiInstanceLoopCharacteristics>
    </userTask>
    <sequenceFlow id="flow2" sourceRef="usertask1" targetRef="usertask2"></sequenceFlow>
    <userTask id="usertask3" name="记录"></userTask>
    <endEvent id="endevent1" name="End"></endEvent>
    <sequenceFlow id="flow4" sourceRef="usertask3" targetRef="endevent1"></sequenceFlow>
    <exclusiveGateway id="exclusivegateway1" name="Exclusive Gateway"></exclusiveGateway>
    <sequenceFlow id="flow7" sourceRef="usertask2" targetRef="exclusivegateway1"></sequenceFlow>
    <sequenceFlow id="flow8" name="通过" sourceRef="exclusivegateway1" targetRef="usertask3">
      <conditionExpression xsi:type="tFormalExpression"><![CDATA[${result == "Y"}]]></conditionExpression>
    </sequenceFlow>
    <sequenceFlow id="flow9" name="拒绝" sourceRef="exclusivegateway1" targetRef="usertask1">
      <conditionExpression xsi:type="tFormalExpression"><![CDATA[${result == "N"}]]></conditionExpression>
    </sequenceFlow>
  </process>
  <bpmndi:BPMNDiagram id="BPMNDiagram_applyHoliday2">
    <bpmndi:BPMNPlane bpmnElement="applyHoliday2" id="BPMNPlane_applyHoliday2">
      <bpmndi:BPMNShape bpmnElement="startevent1" id="BPMNShape_startevent1">
        <omgdc:Bounds height="35.0" width="35.0" x="250.0" y="210.0"></omgdc:Bounds>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="usertask1" id="BPMNShape_usertask1">
        <omgdc:Bounds height="55.0" width="105.0" x="330.0" y="200.0"></omgdc:Bounds>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="usertask2" id="BPMNShape_usertask2">
        <omgdc:Bounds height="55.0" width="105.0" x="500.0" y="200.0"></omgdc:Bounds>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="usertask3" id="BPMNShape_usertask3">
        <omgdc:Bounds height="55.0" width="105.0" x="860.0" y="200.0"></omgdc:Bounds>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="endevent1" id="BPMNShape_endevent1">
        <omgdc:Bounds height="35.0" width="35.0" x="1080.0" y="210.0"></omgdc:Bounds>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape bpmnElement="exclusivegateway1" id="BPMNShape_exclusivegateway1">
        <omgdc:Bounds height="40.0" width="40.0" x="710.0" y="207.0"></omgdc:Bounds>
      </bpmndi:BPMNShape>
      <bpmndi:BPMNEdge bpmnElement="flow1" id="BPMNEdge_flow1">
        <omgdi:waypoint x="285.0" y="227.0"></omgdi:waypoint>
        <omgdi:waypoint x="330.0" y="227.0"></omgdi:waypoint>
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge bpmnElement="flow2" id="BPMNEdge_flow2">
        <omgdi:waypoint x="435.0" y="227.0"></omgdi:waypoint>
        <omgdi:waypoint x="500.0" y="227.0"></omgdi:waypoint>
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge bpmnElement="flow4" id="BPMNEdge_flow4">
        <omgdi:waypoint x="965.0" y="227.0"></omgdi:waypoint>
        <omgdi:waypoint x="1080.0" y="227.0"></omgdi:waypoint>
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge bpmnElement="flow7" id="BPMNEdge_flow7">
        <omgdi:waypoint x="605.0" y="227.0"></omgdi:waypoint>
        <omgdi:waypoint x="710.0" y="227.0"></omgdi:waypoint>
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge bpmnElement="flow8" id="BPMNEdge_flow8">
        <omgdi:waypoint x="750.0" y="227.0"></omgdi:waypoint>
        <omgdi:waypoint x="860.0" y="227.0"></omgdi:waypoint>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="14.0" width="24.0" x="750.0" y="227.0"></omgdc:Bounds>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge bpmnElement="flow9" id="BPMNEdge_flow9">
        <omgdi:waypoint x="730.0" y="247.0"></omgdi:waypoint>
        <omgdi:waypoint x="729.0" y="354.0"></omgdi:waypoint>
        <omgdi:waypoint x="382.0" y="354.0"></omgdi:waypoint>
        <omgdi:waypoint x="382.0" y="255.0"></omgdi:waypoint>
        <bpmndi:BPMNLabel>
          <omgdc:Bounds height="14.0" width="100.0" x="730.0" y="247.0"></omgdc:Bounds>
        </bpmndi:BPMNLabel>
      </bpmndi:BPMNEdge>
    </bpmndi:BPMNPlane>
  </bpmndi:BPMNDiagram>
</definitions>

2.java示例文件

package com.hdkj.activiti;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.activiti.engine.ProcessEngine;
import org.activiti.engine.ProcessEngines;
import org.activiti.engine.RuntimeService;
import org.activiti.engine.repository.Deployment;
import org.activiti.engine.runtime.ProcessInstance;
import org.activiti.engine.runtime.ProcessInstanceQuery;
import org.activiti.engine.task.Task;
import org.junit.Test;
/**
 * @author jh
 * 会签时办理人是否并行在画bpmn图时multi instance中sequential选择false为并行,true为串行
 * 编辑bpmn图,multi instance中completion condition不填为默认所有会签人员全部通过才能完成会签流程,
 * 按比例完成填写如下${nrOfCompletedInstances/nrOfInstances >= 0.5}表示4人中在有人否决前两人通过就完成会签环节
 * 按数量完成填写如下${nrOfCompletedInstances==1}表示4人中在有人否决前一人通过就完成会签环节
 */
public class HuiQian2 {
	
	ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

	/**部署流程定义*/
	@Test
	public void deploymentProcessDefinition(){
		Deployment deployment = processEngine.getRepositoryService()//与流程定义和部署对象相关的Service
						.createDeployment()//创建一个部署对象
						.name("会签流程测试2")//添加部署的名称
						.addClasspathResource("diagrams/HuiQian2.bpmn")//从classpath的资源中加载,一次只能加载一个文件
						.addClasspathResource("diagrams/HuiQian2.png")//从classpath的资源中加载,一次只能加载一个文件
						.deploy();//完成部署
		System.out.println("部署ID:"+deployment.getId());//1
		System.out.println("部署名称:"+deployment.getName());//helloworld入门程序  
	}
	
	/**启动流程实例*/
	@Test
	public void startProcessInstance(){
		//流程定义的key
		String processDefinitionKey = "applyHoliday2";
//		List<String> signList=new ArrayList<String>(); //分配任务的人员
//		signList.add("tom");
//		signList.add("jeck");
//		signList.add("mary");
		//设置会签的办理人
		List<String> signList=Arrays.asList("tom", "jeck", "mary", "lily");
		Map<String, Object> vars = new HashMap<String, Object>(); //参数
		vars.put("signList", signList);
		ProcessInstance pi = processEngine.getRuntimeService()//与正在执行的流程实例和执行对象相关的Service
				.startProcessInstanceByKey(processDefinitionKey, "HuiQian220191205", vars);
//		ProcessInstance pi = processEngine.getRuntimeService()//与正在执行的流程实例和执行对象相关的Service
//				.startProcessInstanceByKey(processDefinitionKey, "HuiQian220191205", null);
		System.out.println("流程实例ID:"+pi.getId());//流程实例ID    101
		System.out.println("流程定义ID:"+pi.getProcessDefinitionId());//流程定义ID   helloworld:1:4
	}
	
	
	/**查询当前人的个人任务*/
	@Test
	public void findMyPersonalTask(){
		//sequential选择false时,会签时大家可以同时办理任务,没有先后顺序
		String assignee = "jeck";
		List<Task> list = processEngine.getTaskService()//与正在执行的任务管理相关的Service
						.createTaskQuery()//创建任务查询对象
						//.taskAssignee(assignee)//指定个人任务查询,指定办理人
						.processInstanceId("7701")//使用流程实例ID查询
						//.taskCandidateGroup(assignee)//组的写法
						.list();
		if(list!=null && list.size()>0){
			for(Task task:list){
				System.out.println("任务ID:"+task.getId());
				System.out.println("任务名称:"+task.getName());
				System.out.println("任务的创建时间:"+task.getCreateTime());
				System.out.println("任务的办理人:"+task.getAssignee());
				System.out.println("流程实例ID:"+task.getProcessInstanceId());
				System.out.println("执行对象ID:"+task.getExecutionId());
				System.out.println("流程定义ID:"+task.getProcessDefinitionId());
				System.out.println("########################################################");
			}
		}
	}
	
	/**完成我的任务*/
	/**
	 * 任务ID根据上个方法可以查询出来,完成后再执行上面的方法,已经查不出来了,说明任务被执行到下一步
	 */
	@Test
	public void completeMyPersonalTask(){
		//任务ID
		String taskId = "8103";
		processEngine.getTaskService()//与正在执行的任务管理相关的Service
					.complete(taskId);
		System.out.println("完成任务:任务ID:"+taskId);
	}
	
	
	/**完成我的任务,message为通过和不通过*/
	@Test
	public void completeMyPersonalTask2(){
		//任务ID
		String taskId = "8012";
		//完成任务的同时,设置流程变量,使用流程变量用来指定完成任务后,下一个连线,对应sequenceFlow.bpmn文件中${message=='不重要'}
		Map<String, Object> variables = new HashMap<String, Object>();
		variables.put("result", "Y");
		processEngine.getTaskService()//与正在执行的任务管理相关的Service
					.complete(taskId,variables);
		System.out.println("完成任务:任务ID:"+taskId);
	}
	/**
	 * 查询整个流程是否完成
	 */
	@Test
	public void isEnd(){
		//第二部的流程实例ID
		String processInstance_Id="7701";
		RuntimeService runtimeService = processEngine.getRuntimeService();
		ProcessInstanceQuery createProcessInstanceQuery = runtimeService.createProcessInstanceQuery();
		ProcessInstanceQuery processInstanceId = createProcessInstanceQuery.processInstanceId(processInstance_Id);
		ProcessInstance singleResult = processInstanceId.singleResult();
		if(singleResult==null){
			System.out.println("执行完毕");
		}else{
			System.out.println("正在执行");
		}
	}

}

 

### 关于工作流的简洁实现 工作流的设计通常涉及多个方面,包括其结构化定义、活动添加以及属性配置。创建一个简单工作流的过程可以被描述为直观且易于理解的方式[^1]。这意味着开发者可以通过清晰的步骤来构建基本的工作流逻辑。 #### 工作流设计的核心原则 1. **模块化** 将复杂的工作分解成更小的任务单元(即活动),这些任务单元可以根据业务需求灵活组合和调整。通过这种方式,能够显著降低整体系统的复杂度并提高可维护性。 2. **标准化支持** 使用行业标准协议和技术栈有助于提升互操作性和扩展能力。例如,在某些Java快速开发框架中集成了Activiti这样的流行工作流引擎,并遵循BPMN (Business Process Model and Notation) 规范[^2]。这不仅简化了XML/JSON文件的一键导入导出功能,还增强了与其他系统的兼容性。 3. **灵活性与定制选项** 提供足够的自定义空间对于满足不同客户的特定需求至关重要。一些先进的解决方案允许用户执行诸如人员动态选择、便捷式会签设置等功能;甚至还可以让用户自行定义表单字段或者控制各个节点上的行为表现形式等等。 4. **面向对象及领域驱动设计理念的应用** openAuth.Core 是另一个很好的例子,该平台基于经典的 DDD 架构模式进行了精心打造——尽管严格意义上讲属于轻量级版本-DDD Lite—但它依然保持高度精炼的同时不失强大功能性[^3] 。这种做法使得应用程序内部组件之间界限分明,职责划分清楚明了,从而进一步促进了代码质量改进(SOLID 原则),最终达到高效管理目标。 综上所述,无论是从技术选型还是实际编码实践角度来看,“简洁”的理念贯穿始终都是成功实施有效工作流管理系统的关键所在之一。 ```python # 示例:简单的Python状态机模拟器用于表示基础工作流概念 class WorkflowState: def __init__(self, name): self.name = name def transition(self, next_state=None): if not isinstance(next_state, WorkflowState): raise ValueError('Invalid state type') print(f'Transitioning from {self.name} to {next_state.name}') return next_state start = WorkflowState('Start') process_task = WorkflowState('Process Task') complete = WorkflowState('Complete') workflow_sequence = start.transition(process_task).transition(complete) print(workflow_sequence.name) ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值