Activiti工作流总结(常用部分)

Activiti工作流

一、工作流的概念

1.生活中常见的工作流

1,请假
2,报销
3,申请转武汉户口
4,购物

2.概述

工作流(Workflow),就是“业务过程的部分或整体在计算机应用环境下的自动化”,它主要解决的是“使在多个参与者之间按照某种预定义的规则传递文档、信息或任务的过程自动进行,从而实现某个预期的业务目标,或者促使此目标的实现”。

工作流管理系统(Workflow Management System, WfMS)是一个软件系统,它完成工作量的定义和管理,并按照在系统中预先定义好的工作流规则进行工作流实例的执行。工作流管理系统不是企业的业务系统,而是为企业的业务系统的运行提供了一个软件的支撑环境。

工作流管理联盟(WfMC,Workflow Management Coalition)给出的关于工作流管理系统的定义是:工作流管理系统是一个软件系统,它通过执行经过计算的流程定义去支持一批专门设定的业务流程。工作流管理系统被用来定义、管理、和执行工作流程。

工作流管理系统的目标:管理工作的流程以确保工作在正确的时间被期望的人员所执行——在自动化进行的业务过程中插入人工的执行和干预。

二、Activiti介绍

1,概述

Activiti5是由Alfresco软件在2010年5月17日发布的业务流程管理(BPM)框架,它是覆盖了业务流程管理、工作流、服务协作等领域的一个开源的、灵活的、易扩展的可执行流程语言框架。Activiti基于Apache许可的开源BPM平台,创始人Tom Baeyens是JBoss jBPM的项目架构师,它特色是提供了eclipse插件,开发人员可以通过插件直接绘画出业务

2,工作流引擎

ProcessEngine对象,这是Activiti工作的核心。负责生成流程运行时的各种实例及数据、监控和管理流程的运行。

3,BPMN

业务流程建模与标注(Business Process Model and Notation,BPMN) ,描述流程的基本符号,包括这些图元如何组合成一个业务流程图(Business Process Diagram)

4,数据库

1)Activiti数据库支持:

Activiti的后台是有数据库的支持,所有的表都以ACT_开头。 第二部分是表示表的用途的两个字母标识。 用途也和服务的API对应。
ACT_RE_: 'RE’表示repository。 这个前缀的表包含了流程定义和流程静态资源 (图片,规则,等等)。
ACT_RU_
: 'RU’表示runtime。 这些运行时的表,包含流程实例,任务,变量,异步任务,等运行中的数据。 Activiti只在流程实例执行过程中保存这些数据, 在流程结束时就会删除这些记录。 这样运行时表可以一直很小速度很快。
ACT_ID_: 'ID’表示identity。 这些表包含身份信息,比如用户,组等等。
ACT_HI_
: 'HI’表示history。 这些表包含历史数据,比如历史流程实例, 变量,任务等等。
ACT_GE_*: 通用数据, 用于不同场景下,如存放资源文件。

2)资源库流程规则表

  1. act_re_deployment 部署信息表
  2. act_re_model 流程设计模型部署表
  3. act_re_procdef 流程定义数据表
    2:运行时数据库表
  4. act_ru_execution 运行时流程执行实例表
  5. act_ru_identitylink 运行时流程人员表,主要存储任务节点与参与者的相关信息
  6. act_ru_task 运行时任务节点表
  7. act_ru_variable 运行时流程变量数据表
    3:历史数据库表
  8. act_hi_actinst 历史节点表
  9. act_hi_attachment 历史附件表
  10. act_hi_comment 历史意见表
  11. act_hi_identitylink 历史流程人员表
  12. act_hi_detail 历史详情表,提供历史变量的查询
  13. act_hi_procinst 历史流程实例表
  14. act_hi_taskinst 历史任务实例表
  15. act_hi_varinst 历史变量表
    4:组织机构表
  16. act_id_group 用户组信息表
  17. act_id_info 用户扩展信息表
  18. act_id_membership 用户与用户组对应信息表
  19. act_id_user 用户信息表
    这四张表很常见,基本的组织机构管理,关于用户认证方面建议还是自己开发一套,组件自带的功能太简单,使用中有很多需求难以满足
    5:通用数据表
  20. act_ge_bytearray 二进制数据表
  21. act_ge_property 属性数据表存储整个流程引擎级别的数据,初始化表结构时,会默认插入三条记录,

3)与Service的对应

#RepositoryService
SELECT * FROM act_ge_bytearray; #二进制文件表
SELECT * FROM act_re_deployment;#流程部署表
SELECT * FROM act_re_procdef;#流程定义
SELECT * FROM act_ge_property;#工作流的ID算法和版本信息表

#RuntimeService TaskService
SELECT * FROM act_ru_execution;#流程启动一次只要没有执行完,就会有一条数据
SELECT * FROM act_ru_task;#可能有多条数据
SELECT * FROM act_ru_variable;#记录流程运行时的流程变量
SELECT * FROM act_ru_identitylink;#存放流程办理人的信息

#HistroyService
SELECT * FROM act_hi_procinst;#历史流程实例
SELECT * FROM act_hi_taskinst;#历史任务实例
SELECT * FROM act_hi_actinst;#历史活动节点表
SELECT * FROM act_hi_varinst;#历史流程变量表
SELECT * FROM act_hi_identitylink;##历史办理人表
SELECT * FROM act_hi_comment;#批注表
SELECT * FROM act_hi_attachment;#附件表

#IdentityService
SELECT * FROM act_id_group #角色
SELECT * FROM act_id_membership#用户和角色之间的关系
SELECT * FROM act_id_info#用户的详细信息
SELECT * FROM act_id_user#用户表

三、Activiti5开发环境

1、准备工作

1.创建项目

2.配置pom.xml

3.配置log4j.properties

2、初始化数据

1).创建数据库

2).创建类去初始化表方式一

	@Test
	public void initTables() {
		//创建数据源
		DriverManagerDataSource dataSource=new DriverManagerDataSource();
		dataSource.setDriverClassName("com.mysql.jdbc.Driver");
		dataSource.setUrl("jdbc:mysql://127.0.0.1:3306/0705activiti");
		dataSource.setUsername("root");
		dataSource.setPassword("123456");
		
		// 创建流程引擎的配置
		ProcessEngineConfiguration configuration = ProcessEngineConfiguration
				.createStandaloneProcessEngineConfiguration();
		configuration.setJdbcDriver("com.mysql.jdbc.Driver");
		configuration.setJdbcUrl("jdbc:mysql://127.0.0.1:3306/0705activiti");
		configuration.setJdbcUsername("root");
		configuration.setJdbcPassword("123456");
//		configuration.setDataSource(dataSource);
		/** 
		 * ProcessEngineConfiguration.DB_SCHEMA_UPDATE_FALSE  如果数据库里面没有activit的表,也不会创建
		 * ProcessEngineConfiguration.DB_SCHEMA_UPDATE_CREATE_DROP 创建表,使用完之后删除
		 * ProcessEngineConfiguration.DB_SCHEMA_UPDATE_TRUE  如果数据库里面没有表,就创建
		 * 
		 * dorp-create 代表如果数据库里面有表,那么先删除再创建
		 * 
		 */
		//配置表的初始化的方式
		configuration.setDatabaseSchemaUpdate(ProcessEngineConfiguration.DB_SCHEMA_UPDATE_TRUE);
		
		//得到流程引擎
		ProcessEngine processEngine=configuration.buildProcessEngine();
		System.out.println(processEngine);
	}

3).创建类去初始化表方式二

创建activiti.cfg.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
	xmlns:aop="http://www.springframework.org/schema/aop" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://www.springframework.org/schema/beans 
	http://www.springframework.org/schema/beans/spring-beans.xsd
	http://www.springframework.org/schema/context
	http://www.springframework.org/schema/context/spring-context.xsd
	http://www.springframework.org/schema/tx
	http://www.springframework.org/schema/tx/spring-tx.xsd
	http://www.springframework.org/schema/aop
	http://www.springframework.org/schema/aop/spring-aop.xsd">
	
	
   <bean id="processEngineConfiguration" class="org.activiti.engine.impl.cfg.StandaloneProcessEngineConfiguration">
   		<property name="jdbcDriver" value="com.mysql.jdbc.Driver"></property>
   		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/0705activiti"></property>
   		<property name="jdbcUsername" value="root"></property>
   		<property name="jdbcPassword" value="123456"></property>
		<!--    		
		flase: 默认值。activiti在启动时,会对比数据库表中保存的版本,如果没有表或者版本不匹配,将抛出异常。
		true: activiti会对数据库中所有表进行更新操作。如果表不存在,则自动创建。
		create_drop: 在activiti启动时创建表,在关闭时删除表(必须手动关闭引擎,才能删除表)。
		drop-create: 在activiti启动时删除原来的旧表,然后在创建新表(不需要手动关闭引擎)。 -->
   		<property name="databaseSchemaUpdate" value="drop-create"></property>
   </bean>
</beans>

创建方法


	@Test
	public void intiTables2() {
		ProcessEngineConfiguration configuration = ProcessEngineConfiguration
				.createProcessEngineConfigurationFromResource("/activiti.cfg.xmls");
		// 得到流程引擎
		ProcessEngine processEngine = configuration.buildProcessEngine();
		System.out.println(processEngine);

	}

4).创建类去初始化表方式三

	@Test
	public void intiTables3() {
		//必须创建activiti.cfg.xml  并配置好数据库的信息
		ProcessEngine processEngine=ProcessEngines.getDefaultProcessEngine();
		System.out.println(processEngine);
	}

四、activiti.cfg.xml的说明

1,Activiti核心配置文件,配置流程引擎创建工具的基本参数和数据库连接池参数。

定义数据库配置参数:
1,jdbcUrl: 数据库的JDBC URL。
2,jdbcDriver: 对应不同数据库类型的驱动。
3,jdbcUsername: 连接数据库的用户名。
4,jdbcPassword: 连接数据库的密码。
基于JDBC参数配置的数据库连接 会使用默认的MyBatis连接池。 下面的参数可以用来配置连接池(来自MyBatis参数):
1,jdbcMaxActiveConnections: 连接池中处于被使用状态的连接的最大值。默认为10。
2,jdbcMaxIdleConnections: 连接池中处于空闲状态的连接的最大值。
3,jdbcMaxCheckoutTime: 连接被取出使用的最长时间,超过时间会被强制回收。 默认为20000(20秒)。
4,jdbcMaxWaitTime: 这是一个底层配置,让连接池可以在长时间无法获得连接时, 打印一条日志,并重新尝试获取一个连接。(避免因为错误配置导致沉默的操作失败)。 默认为20000(20秒)。
5,dataSource 可以配置其它数据源
6,databaseSchemaUpdate 设置数据库表的创建策略

2,示例数据库配置:

在这里插入图片描述

3,使用其它数据源:

在这里插入图片描述

五、核心API

ProcessEngine
说明:
1)在Activiti中最核心的类,其他的类都是由他而来。
2)产生方式。

1,ProcessEngine获取方式

	@Test
	public void intiTables3() {
		//必须创建activiti.cfg.xml  并配置好数据库的信息
		ProcessEngine processEngine=ProcessEngines.getDefaultProcessEngine();
		System.out.println(processEngine);
		//流程图的部署  修改  删除的服务器 act_ge_bytearray, act_re_deployment, act_re_model, act_re_procdef
		RepositoryService repositoryService = processEngine.getRepositoryService();
		//流程的运行 act_ru_event_subscr, act_ru_execution, act_ru_identitylink, act_ru_job, act_ru_task, act_ru_variable
		RuntimeService runtimeService = processEngine.getRuntimeService();
		TaskService taskService = processEngine.getTaskService();
		//查询历史记录的服务器act_hi_actinst, act_hi_attachment, act_hi_comment, act_hi_detail, act_hi_identitylink, act_hi_procinst, act_hi_taskinst, act_hi_varinst
		HistoryService historyService = processEngine.getHistoryService();
		//页面表单的服务器[了解]
		FormService formService = processEngine.getFormService();
		//对工作流的用户管理的表的操作act_id_group, act_id_info, act_id_membership, act_id_user
		IdentityService identityService = processEngine.getIdentityService();
		//管理器
		ManagementService managementService = processEngine.getManagementService();
	}

2,RepositoryService

是Activiti的仓库服务类。所谓的仓库指流程定义文档的两个文件,bpmn文件和流程图片。
1)产生方式

2)可以产生DeploymentBuilder,用来定义流程部署的相关参数

3)删除流程定义

3,RuntimeService

是activiti的流程执行服务类。可以从这个服务类中获取很多关于流程执行相关的信息。

4,TaskService

是activiti的任务服务类。可以从这个类中获取任务的信息。

5,HistoryService

是activiti的查询历史信息的类。在一个流程执行完成后,这个对象为我们提供查询历史信息。

6,ProcessDefinition act_re_procdef

流程定义类。可以从这里获得资源文件等。当流程图被部署之后,查询出来的数据就是流程定义的数据

7,ProcessInstance act_ru_execution

代表流程定义的执行实例。如范冰冰请了一天的假,她就必须发出一个流程实例的申请。一个流程实例包括了所有的运行节点。我们可以利用这个对象来了解当前流程实例的进度等信息。流程实例就表示一个流程从开始到结束的最大的流程分支,即一个流程中流程实例只有一个。

8,Execution act_ru_execution

Activiti用这个对象去描述流程执行的每一个节点。在没有并发的情况下,Execution就是同ProcessInstance。流程按照流程定义的规则执行一次的过程,就可以表示执行对象Execution。
如图为ProcessInstance的源代码,

9,TaskInstance 任务实例act_ru_task

六 HelloWorld程序(模拟流程的执行)

1.画流程图

1)画流程图

1.画流程图

在这里插入图片描述

2).设置任务的办理人

在这里插入图片描述

在这里插入图片描述

3).设置流程的ID和NAME

在这里插入图片描述

2.部署流程定义

在这里插入图片描述

3.启动流程

在这里插入图片描述

4.查询任务

在这里插入图片描述

5.办理任务

在这里插入图片描述

七、管理流程定义

功能:对流程的增加 修改 删除 查询
act_ge_bytearray act_re_deployment act_re_procdef

1.流程图

在这里插入图片描述

2.部署流程定义【两种方式 classpath zip】

//得到流程引擎
	private ProcessEngine processEngine=ProcessEngines.getDefaultProcessEngine();
	
	/**
	 * 部署流程使用classpath
	 */
	@Test
	public void deployProcess01() {
		//得到流程部署的service
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		Deployment deploy = repositoryService.createDeployment().name("请假流程001")
		.addClasspathResource("HelloWorld.bpmn")
		.addClasspathResource("HelloWorld.png")
		.deploy();
		System.out.println("部署成功:流程部署ID:"+deploy.getId());
	}
	
	/**
	 * 部署流程使用zip
	 * 流程图的文件必须是xxxx.zip结尾
	 */
	@Test
	public void deployProcess02() {
		//如果不加/代表从当前包里面找文件 
		InputStream inputStream = this.getClass().getResourceAsStream("/HelloWorld.zip");
		//如果加/代表从classpath的根目录里面找文件
		//InputStream inputStream = this.getClass().getResourceAsStream("/HelloWorld.zip");
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		ZipInputStream zipInputStream =new ZipInputStream(inputStream);
		Deployment deploy = repositoryService.createDeployment().name("请假流程001")
		.addZipInputStream(zipInputStream)//添加流程图的流
		.deploy();//确定部署
		System.out.println("部署成功,部署ID:"+deploy.getId());
	}
	

3.查询部署信息act_re_deployment

/**
	 * 查询流程部署信息  act_re_deployment
	 */
	@Test
	public void queryProcessDeploy () {
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		//创建部署信息的查询
		String deploymentId="1";
//		Deployment deploy = repositoryService.createDeploymentQuery()
		List<Deployment> list= repositoryService.createDeploymentQuery()
		//条件
		//.deploymentId(deploymentId)  //根据部署ID去查询
		//.deploymentName(name)//根据部署名称去查询
		//.deploymentTenantId(tenantId)//根据tenantId去查询
//		.deploymentNameLike(nameLike)//根据部署名称模糊查询
		//.deploymentTenantIdLike(tenantIdLike)//根据tenantId模糊查询
		//排序
//		.orderByDeploymentId().asc()  //根据部署ID升序
		//.orderByDeploymenTime().desc() //根据部署时间降序
		//.orderByDeploymentName()//根据部署名称升序
		//结果集
		.list();  //查询返回list集合
//		.listPage(firstResult, maxResults)  分页查询返回list集合
		//.singleResult(); //返回单个对象
//		.count();
		
		/*System.out.println("部署ID:"+deploy.getId());
		System.out.println("部署名称:"+deploy.getName());
		System.out.println("部署时间:"+deploy.getDeploymentTime());*/
//		System.out.println(count);
		for (Deployment deployment : list) {
			System.out.println("部署ID:"+deployment.getId());
			System.out.println("部署名称:"+deployment.getName());
			System.out.println("部署时间:"+deployment.getDeploymentTime());
			System.out.println("########################");
		}
	}


4.查询流程定义信息act_re_procdef


	/**
	 * 查询流程定义
	 */
	@Test
	public void queryProcDef() {
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		List<ProcessDefinition> list = repositoryService.createProcessDefinitionQuery()
		//条件
//		.deploymentId(deploymentId) 根据部署ID查询
//		.deploymentIds(deploymentIds) 根据部署ID的集合查询Set<String> deploymentIds
//		.processDefinitionId(processDefinitionId)//根据流程定义IDHelloWorld:1:4
//		.processDefinitionIds(processDefinitionIds)//根据流程定义的IDS查询
//		.processDefinitionKey(processDefinitionKey)//根据流程定义的的key查询
//		.processDefinitionKeyLike(processDefinitionKeyLike)//根据流程定义的的key模糊查询
//		.processDefinitionName(processDefinitionName)//根据流程定义的名称查询
//		.processDefinitionNameLike(processDefinitionNameLike)//根据流程定义的名称模糊查询
//		.processDefinitionResourceName(resourceName)//根据流程图的BPMN文件名查询
//		.processDefinitionResourceNameLike(resourceNameLike)//根据流程图的BPMN文件名模糊查询
//		.processDefinitionVersion(processDefinitionVersion)//根据流程定义的版本查询
//		.processDefinitionVersionGreaterThan(processDefinitionVersion)//version>num
//		.processDefinitionVersionGreaterThanOrEquals(processDefinitionVersion)//version>=num
//		.processDefinitionVersionLowerThan(processDefinitionVersion)//version<num
//		.processDefinitionVersionLowerThanOrEquals(processDefinitionVersion)//version<=num
		//排序
//		.orderByDeploymentId()
//		.orderByProcessDefinitionId()
//		.orderByProcessDefinitionKey()
//		.orderByProcessDefinitionName()
//		.orderByProcessDefinitionVersion()
		//结果集
		.list();
//		.listPage(firstResult, maxResults)\
//		.count()
//		.singleResult()
		if(null!=list&&list.size()>0) {
			for (ProcessDefinition pd : list) {
				System.out.println("流程定义ID:"+pd.getId());
				System.out.println("流程部署ID:"+pd.getDeploymentId());
				System.out.println("流程定义KEY:"+pd.getKey());
				System.out.println("流程定义的名称:"+pd.getName());
				System.out.println("流程定义的bpmn文件名:"+pd.getResourceName());//bpmn的name
				System.out.println("流程图片名:"+pd.getDiagramResourceName());//png的name
				System.out.println("流程定义的版本号:"+pd.getVersion());
				System.out.println("##################");
			}
		}
		
	}

5.删除流程定义信息

	/**
	 * 删除流程定义
	 */
	@Test
	public void deleteProcessDef() {
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		String deploymentId="2501";
		//根据流程部署id删除流程定义 如果当前id的流程正在执行,那么会报错
		//repositoryService.deleteDeployment(deploymentId);
		//根据流程部署id删除删除流程定义 如果当前id的流程正在执行,会把正在执行的流程数据删除 act_ru_*和act_hi_*表里面的数据
		repositoryService.deleteDeployment(deploymentId, true);
//		repositoryService.deleteDeploymentCascade(deploymentId);==repositoryService.deleteDeployment(deploymentId, true);
		System.out.println("删除成功");
	}

6.修改流程定义信息

修改流程图之后重新部署,只要key不变,它的版本号就会+1

在这里插入图片描述

7.查询流程图

	/**
	 * 查询流程图  根据流程定义ID
	 */
	@Test
	public void viewProcessImg() {
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		String processDefinitionId="HelloWorld:1:4";
		InputStream inputStream = repositoryService.getProcessDiagram(processDefinitionId);
		
		File file=new File("d:/HelloWorld.png");
		try {
			BufferedOutputStream outputStream=new BufferedOutputStream(new FileOutputStream(file));
			int len=0;
			byte[] b=new byte[1024];
			while((len=inputStream.read(b))!=-1) {
				outputStream.write(b, 0, len);
				outputStream.flush();
			}
			outputStream.close();
			inputStream.close();
			System.out.println("查询成功");
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
	/**
	 * 查询流程图  根据流流程部署ID
	 */
	@Test
	public void viewProcessImg2() {
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		//根据流程部署ID查询流程定义对象
		String deploymentId="1";
		ProcessDefinition processDefinition = repositoryService.createProcessDefinitionQuery().deploymentId(deploymentId).singleResult();
		//从流程定义对象里面查询出流程定义ID
		String processDefinitionId=processDefinition.getId();
		InputStream inputStream = repositoryService.getProcessDiagram(processDefinitionId);
		
		File file=new File("d:/"+processDefinition.getDiagramResourceName());
		try {
			BufferedOutputStream outputStream=new BufferedOutputStream(new FileOutputStream(file));
			int len=0;
			byte[] b=new byte[1024];
			while((len=inputStream.read(b))!=-1) {
				outputStream.write(b, 0, len);
				outputStream.flush();
			}
			outputStream.close();
			inputStream.close();
			System.out.println("查询成功");
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}

8.附加功能:查询最新版本的流程定义

	/**
	 * 查询最新的流程定义
	 */
	@Test
	public void queryNewProcessDef() {
		Map<String, ProcessDefinition> map = new HashMap<>();

		// 查询所有的流程定义根据版本号升序
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		List<ProcessDefinition> list = repositoryService.createProcessDefinitionQuery()
				.orderByProcessDefinitionVersion().asc().list();
		if(null!=list&&list.size()>0) {
			for (ProcessDefinition pd : list) {
				map.put(pd.getKey(), pd);
			}
		}
		//循环map集合
		Collection<ProcessDefinition> values = map.values();
		for (ProcessDefinition pd : values) {
			System.out.println("流程定义ID:" + pd.getId());
			System.out.println("流程部署ID:" + pd.getDeploymentId());
			System.out.println("流程定义KEY:" + pd.getKey());
			System.out.println("流程定义的名称:" + pd.getName());
			System.out.println("流程定义的bpmn文件名:" + pd.getResourceName());// bpmn的name
			System.out.println("流程图片名:" + pd.getDiagramResourceName());// png的name
			System.out.println("流程定义的版本号:" + pd.getVersion());
			System.out.println("##################");
		}
	}


9.附加功能:删除流程定义(删除key相同的所有不同版本的流程定义)

	/**
	 * 已知key 附加功能:删除流程定义(删除key相同的所有不同版本的流程定义)
	 */
	public void deleteAllSameVersion() {
		String processDefinitionKey = "HelloWorld";
		RepositoryService repositoryService = this.processEngine.getRepositoryService();
		// 根据流程定义的key查询流程集合
		List<ProcessDefinition> list = repositoryService.createProcessDefinitionQuery()
				.processDefinitionKey(processDefinitionKey).list();

		if (null != list && list.size() > 0) {
			for (ProcessDefinition pd : list) {
				repositoryService.deleteDeployment(pd.getDeploymentId(), true);

			}
		}

	}

八、流程实例、任务的执行

1,流程图

在这里插入图片描述

2,部署流程定义

3,启动流程实例

	/**
	 * 启动流程
	 */
	@Test
	public void startProcess() {
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
//		runtimeService.startProcessInstanceById(processDefinitionId)//根据流程定义ID启动流程
		/**
		 * 参数1:流程定义ID
		 * 参数2:Map<String,Object> 流程变量
		 */
//		runtimeService.startProcessInstanceById(processDefinitionId, variables);
		/**
		 * 参数1:流程定义ID
		 * 参数2:String 业务ID 把业务ID和流程执行实例进行绑定
		 */
//		runtimeService.startProcessInstanceById(processDefinitionId, businessKey);
		/**
		 * 参数1:流程定义ID
		 * 参数2:String 业务ID 把业务ID和流程执行实例进行绑定
		 * 参数3:Map<String,Object> 流程变量
		 */
//		runtimeService.startProcessInstanceById(processDefinitionId, businessKey, variables)
//		
//		runtimeService.startProcessInstanceByKey(processDefinitionKey)//根据流程定义的key启动
		/**
		 * 参数1:流程定义的Key
		 * 参数2:Map<String,Object> 流程变量
		 */
//		runtimeService.startProcessInstanceByKey(processDefinitionKey, variables)
		/**
		 * 参数1:流程定义Key
		 * 参数2:String 业务ID 把业务ID和流程执行实例进行绑定
		 */
//		runtimeService.startProcessInstanceByKey(processDefinitionId, businessKey);
		/**
		 * 参数1:流程定义Key
		 * 参数2:String 业务ID 把业务ID和流程执行实例进行绑定
		 * 参数3:Map<String,Object> 流程变量
		 */
//		runtimeService.startProcessInstanceByKey(processDefinitionId, businessKey, variables)
		
		//实例开发中使用的
		//runtimeService.startProcessInstanceByKey(processDefinitionId, businessKey, variables)
		//runtimeService.startProcessInstanceByKey(processDefinitionId, businessKey);
		
		String processDefinitionKey="HelloWorld";
		ProcessInstance processInstance = runtimeService.startProcessInstanceByKey(processDefinitionKey);
		System.out.println("流程启动成功:"+processInstance.getId()+"   "+processInstance.getProcessDefinitionId()+"  "+processInstance.getProcessInstanceId());
		
	}

4,查询我的个人任务

	/**
	 * 查询我的个人任务act_ru_task
	 */
	@Test
	public void queryMyTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String assignee="张三";
		List<Task> list = taskService.createTaskQuery()
		//条件
		.taskAssignee(assignee)//根据任务办理人查询任务
//		.deploymentId(deploymentId)//根据部署ID查询 where id=id
//		.deploymentIdIn(deploymentIds)//根据部署ID集合查询   where id in (1,2,3,4)
//		.executionId(executionId)//根据执行实例ID
//		.processDefinitionId(processDefinitionId)//根据流程定义ID
//		.processDefinitionKey(processDefinitionKey)//根据流程定义的key
//		.processDefinitionKeyIn(processDefinitionKeys)
//		.processDefinitionKeyLike(processDefinitionKeyLike)
//		.processDefinitionName(processDefinitionName)
//		.processDefinitionNameLike(processDefinitionNameLike)
//		.processInstanceBusinessKey(processInstanceBusinessKey)
		//排序
		.orderByTaskCreateTime().desc()
		//结果集
		.list();
//		.listPage(firstResult, maxResults)
//		.count();
//		.singleResult()
		
		if(null!=list&&list.size()>0) {
			for (Task task : list) {
				System.out.println("任务ID:"+task.getId());
				System.out.println("任务办理人:"+task.getAssignee());
				System.out.println("执行实例ID:"+task.getExecutionId());
				System.out.println("任务名称:"+task.getName());
				System.out.println("流程定义ID:"+task.getProcessDefinitionId());
				System.out.println("流程实例ID:"+task.getProcessInstanceId());
				System.out.println("任务创建时间:"+task.getCreateTime());
				System.out.println("####################");
			}
		}
	}
	
	

5,办理任务

	/**
	 * 办理任务
	 */
	@Test
	public void completeTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String taskId="2504";
		//根据任务ID去完成任务
		taskService.complete(taskId);
		//根据任务ID去完成任务并指定流程变量
//		taskService.complete(taskId, variables);
		System.out.println("任务完成");
	}
	

6,查询流程状态(判断流程正在执行,还是结束)

	/**
	 * 判断流程是否结束 作用:更新业务表里面的状态
	 */
	@Test
	public void isComplete() {
		// 已知流程实例ID
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
		String processInstanceId = "2501";
		ProcessInstance processInstance = runtimeService.createProcessInstanceQuery()
				.processInstanceId(processInstanceId).singleResult();
		if (null != processInstance) {
			System.out.println("流程未结束");
		} else {
			System.out.println("流程已结束");
		}

		//已知任务ID 5002 [了解]
		//根据任务ID查询任务实例对象
		/*TaskService taskService = this.processEngine.getTaskService();
		String taskId="5002";
		Task task = taskService.createTaskQuery().taskId(taskId).singleResult();
		//从任务实例里面取出流程实例ID
		String processInstanceId2 = task.getProcessInstanceId();
		//使用流程实例ID去流程实例表里面查询有没有数据
		ProcessInstanceQuery processInstance2 = runtimeService.createProcessInstanceQuery().processInstanceId(processInstanceId2);
		if (null != processInstance2) {
			System.out.println("流程未结束");
		} else {
			System.out.println("流程已结束");
		}*/
		
		
		
	}

7,查询正在执行的流程实例

	/**
	 * 7,查询当前的流程实例 act_ru_execution
	 * 
	 */
	@Test
	public void queryProcessInstance() {
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
		List<ProcessInstance> list = runtimeService.createProcessInstanceQuery().list();
		if(null!=list&&list.size()>0) {
			for (ProcessInstance pi : list) {
				System.out.println("执行实例ID:"+pi.getId());
				System.out.println("流程定义ID:"+pi.getProcessDefinitionId());
				System.out.println("流程实例ID:"+pi.getProcessInstanceId());
				System.out.println("########################");
			}
		}
	}

8,总结

1)Execution 执行对象

按流程定义的规则执行一次的过程.
对应的表:
act_ru_execution, 正在执行的信息
act_hi_procinst,已经执行完的历史流程实例信息
act_hi_actinst,存放历史所有完成的活动

2)ProcessInstance 流程实例

特指流程从开始到结束的那个最大的执行分支,一个执行的流程中,流程实例只有1个。
注意
(1)如果是单例流程,执行对象ID就是流程实例ID
(2)如果一个流程有分支和聚合,那么执行对象ID和流程实例ID就不相同
(3)一个流程中,流程实例只有1个,执行对象可以存在多个。

3)Task 任务

执行到某任务环节时生成的任务信息。
对应的表:
act_ru_task,正在执行的任务信息
act_hi_taskinst,已经执行完的历史任务信息

4)相关ID总结

部署ID —act_re_deployment id
流程定义ID act_re_procdef id
流程实例ID act_ru_execution id
执行实例ID act_ru_task execution_id
任务ID act_run_task id

九、流程变量

1:流程图

在这里插入图片描述

流程变量在整个工作流中扮演很重要的作用。例如:请假流程中有请假天数、请假原因等一些参数都为流程变量的范围。流程变量的作用域范围是只对应一个流程实例。也就是说各个流程实例的流程变量是不相互影响的。流程实例结束完成以后流程变量还保存在数据库中(存放到流程变量的历史表中)。

设置流程变量的值

存在形式 key —value

2:部署流程定义

3:启动流程实例并设置流程变量

	/**
	 * 启动流程
	 */
	@Test
	public void startProcess() {
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
		String processDefinitionKey = "HelloWorld";
//		ProcessInstance processInstance = runtimeService.startProcessInstanceByKey(processDefinitionKey);
		//创建流程变量对象
		Map<String,Object> variables=new HashMap<>();
		variables.put("请假天数", 5);//int
		variables.put("请假原因", "约会");
		variables.put("请假时间", new Date());
		ProcessInstance processInstance = runtimeService.startProcessInstanceByKey(processDefinitionKey, variables);
		System.out.println("流程启动成功:" + processInstance.getId() + "   " + processInstance.getProcessDefinitionId() + "  "
				+ processInstance.getProcessInstanceId());

	}

4:设置流程变量

	/**
	 * 设置流程变量1
	 */
	@Test
	public void setVariables() {
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
		String executionId="2501";
		//runtimeService.setVariable(executionId, "请假人", "小明");
		Map<String,Object> variables=new HashMap<>();
		variables.put("请假天数", 6);//int
		variables.put("请假原因", "约会妹子");
		variables.put("请假时间", new Date());
		variables.put("用户对象", new User(1,"小明"));
		runtimeService.setVariables(executionId, variables);
		System.out.println("流程变量设置成功");
		
		
	}
	/**
	 * 设置流程变量2
	 */
	@Test
	public void setVariables2() {
		TaskService taskService = this.processEngine.getTaskService();
		
		String taskId="2507";
		//runtimeService.setVariable(executionId, "请假人", "小明");
		Map<String,Object> variables=new HashMap<>();
		variables.put("任务ID设置的", 9);//int
//		taskService.setVariable(taskId, variableName, value);
		taskService.setVariables(taskId, variables);
		System.out.println("流程变量设置成功");
	}

5:获取流程变量

/**
	 * 获取流程变量
	 */
	@Test
	public void getVariables() {
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
		String executionId="2501";
		Integer days=(Integer) runtimeService.getVariable(executionId, "请假天数");
		Date date=(Date) runtimeService.getVariable(executionId, "请假时间");
		User user=(User) runtimeService.getVariable(executionId, "用户对象");
		System.out.println(days);
		System.out.println(date.toLocaleString());
		System.out.println(user.getId()+"  "+user.getName());
		

	}

6:流程变量的支持的类型

在这里插入图片描述

7:总结

1):流程变量

在流程执行或者任务执行的过程中,用于设置和获取变量,使用流程变量在流程传递的过程中传递业务参数。
对应的表:
act_ru_variable:正在执行的流程变量表
act_hi_varinst:流程变量历史表

2):扩展知识:setVariable和setVariableLocal的区别

setVariable:设置流程变量的时候,流程变量名称相同的时候,后一次的值替换前一次的值,而且可以看到TASK_ID的字段不会存放任务ID的值
setVariableLocal:
1:设置流程变量的时候,针对当前活动的节点设置流程变量,如果一个流程中存在2个活动节点,对每个活动节点都设置流程变量,即使流程变量的名称相同,后一次的版本的值也不会替换前一次版本的值,它会使用不同的任务ID作为标识,存放2个流程变量值,而且可以看到TASK_ID的字段会存放任务ID的值 例如act_hi_varinst 表的数据:不同的任务节点,即使流程变量名称相同,存放的值也是不同的。
如图:

2:还有,使用setVariableLocal说明流程变量绑定了当前的任务,当流程继续执行时,下个任务获取不到这个流程变量(因为正在执行的流程变量中没有这个数据),所有查询正在执行的任务时不能查询到我们需要的数据,此时需要查询历史的流程变量。

十、流程执行历史记录

1,查询历史流程实例

	/* 1,查询历史流程实例 */
	@Test
	public void historyProcessInstince() {
		List<HistoricProcessInstance> list = historyService.createHistoricProcessInstanceQuery()
		//条件
//		.processDefinitionId(processDefinitionId)
//		.processDefinitionKey(processDefinitionKey)
//		.processDefinitionKeyIn(processDefinitionKeys)
//		.processDefinitionName(processDefinitionName)
//		.processDefinitionVersion(processDefinitionVersion)
//		.processInstanceBusinessKey(processInstanceBusinessKey)
//		.processInstanceId(processInstanceId)
//		.processInstanceIds(processInstanceIds)
		//排序
//		.orderByProcessDefinitionId()
//		.orderByProcessInstanceBusinessKey()
//		.orderByProcessInstanceDuration()
//		.orderByProcessInstanceStartTime()
//		.orderByProcessInstanceId()
		//结果集
		.list();
//		.listPage(firstResult, maxResults)
//		.count()
//		.singleResult();
		
		if(null!=list&&list.size()>0) {
			for (HistoricProcessInstance hpi : list) {
				System.out.println("历史流程实例ID:" + hpi.getId());
				System.out.println("流程定义ID:" + hpi.getProcessDefinitionId());
				System.out.println("历史流程实例的业务ID:" + hpi.getBusinessKey());
				System.out.println("流程部署ID:" + hpi.getDeploymentId());
				System.out.println("流程定义KEY:" + hpi.getProcessDefinitionKey());
				System.out.println("开始活动ID:" + hpi.getStartActivityId());
				System.out.println("结束活动ID:" + hpi.getEndActivityId());
				System.out.println("########################");
			}
		}
		
	}

2,查询历史活动

	/* 2,查询历史活动 */
	@Test
	public void queryHistoryAct() {
		List<HistoricActivityInstance> list = this.historyService.createHistoricActivityInstanceQuery()
		//条件
//		.activityId(activityId)
//		.activityInstanceId(activityInstanceId)
//		.activityName(activityName)
		//排序
//		.orderByActivityId()
//		.orderByActivityName()
		//结果集
		.list();
		if(null!=list&&list.size()>0)
		{
			for (HistoricActivityInstance hai : list) {
				System.out.println("ID:"+hai.getId());
				System.out.println("流程定义ID:"+hai.getProcessDefinitionId());
				System.out.println("流程实例ID:"+hai.getProcessInstanceId());
				System.out.println("执行实例ID:"+hai.getExecutionId());
				System.out.println("活动ID:"+hai.getActivityId());
				System.out.println("任务ID:"+hai.getTaskId());
				System.out.println("活动名称:"+hai.getActivityName());
				System.out.println("活动类型:"+hai.getActivityType());
				System.out.println("任务办理人:"+hai.getAssignee());
				System.out.println("开始时间:"+hai.getStartTime());
				System.out.println("结束时间:"+hai.getEndTime());
				System.out.println("持续时间:"+hai.getDurationInMillis());
				System.out.println("#######################################");
			}
		}
	}

3,查询历史任务

/* 3,查询历史任务 act_hi_taskinst */
	@Test
	public void queryHistoryTask() {
		List<HistoricTaskInstance> list = this.historyService.createHistoricTaskInstanceQuery()
		//条件
//		.deploymentId(deploymentId)
//		.deploymentIdIn(deploymentIds)
//		.executionId(executionId)
//		.processDefinitionId(processDefinitionId)
//		.processDefinitionKey(processDefinitionKey)
//		.processDefinitionKeyIn(processDefinitionKeys)
//		.processDefinitionKeyLike(processDefinitionKeyLike)     processDefinitionKeyLike="%Hello%"
//		.processDefinitionName(processDefinitionName)
//		.processDefinitionNameLike(processDefinitionNameLike)
		//排序
//		.orderByTaskDefinitionKey()
		//结果集
		.list();
//		.listPage(firstResult, maxResults)
//		.count()
//		.singleResult()
		if(null!=list&&list.size()>0)
		{
			for (HistoricTaskInstance task : list) {
				System.out.println("任务ID:" + task.getId());
				System.out.println("任务办理人:" + task.getAssignee());
				System.out.println("执行实例ID:" + task.getExecutionId());
				System.out.println("任务名称:" + task.getName());
				System.out.println("流程定义ID:" + task.getProcessDefinitionId());
				System.out.println("流程实例ID:" + task.getProcessInstanceId());
				System.out.println("任务创建时间:" + task.getCreateTime());
				System.out.println("任务结束时间:" + task.getEndTime());
				System.out.println("#######################################");
			}
		}
	}

4,查询历史流程变量

/**
	 * 7:查询历史的流程变量
	 */
	@Test
	public void getHistoryVariables() {
		HistoryService historyService = this.processEngine.getHistoryService();
		
		/*HistoricVariableInstance singleResult = historyService.createHistoricVariableInstanceQuery().id("2503").singleResult();;
		System.out.println(singleResult.getId());
		System.out.println(singleResult.getValue());
		System.out.println(singleResult.getVariableName());
		System.out.println(singleResult.getVariableTypeName());*/
		String processInstanceId="2501";
		List<HistoricVariableInstance> list = historyService.createHistoricVariableInstanceQuery().processInstanceId(processInstanceId).list();
		
		for (HistoricVariableInstance hvs : list) {
			System.out.println("ID"+hvs.getId());
			System.out.println("变量值"+hvs.getValue());
			System.out.println("变量名"+hvs.getVariableName());
			System.out.println("变量类型"+hvs.getVariableTypeName());
			System.out.println("#####################");
		}
	}
	

十一、连线

使用流程变量去控制流程的走向

1,流程图

在这里插入图片描述

提交申请节点

在这里插入图片描述

部门经理审批节点

在这里插入图片描述

部门经理审批节点连线

在这里插入图片描述

总经理审批节点

在这里插入图片描述

2,部署流程定义+启动流程实例

3,查询我的个人任务

4,完成任务

/**
	 * 办理任务
	 */
	@Test
	public void completeTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String taskId = "15003";
		// 根据任务ID去完成任务
		taskService.complete(taskId);
		// 根据任务ID去完成任务并指定流程变量
//		taskService.complete(taskId, variables);
		System.out.println("任务完成");
	}
	/**
	 * 办理任务并使用流程变量指定流程走向
	 */
	@Test
	public void completeTask2() {
		TaskService taskService = this.processEngine.getTaskService();
		String taskId = "12502";
		Map<String, Object> variables=new HashMap<>();
		variables.put("outcome", "重要");
		// 根据任务ID去完成任务并指定流程变量
		taskService.complete(taskId, variables);
		System.out.println("任务完成");
	}

5,总结

1)、一个活动中可以指定一个或多个SequenceFlow(Start中有一个,End中没有)。

  • 开始活动中有一个SequenceFlow (即连线)。
  • 结束活动中没有SequenceFlow 。
  • 其他活动中有1条或多条SequenceFlow

2)、如果只有一个,则可以不使用流程变量设置codition的名称;

在这里插入图片描述

3)如果有多个,则需要使用流程变量设置codition的名称。

outcomes表示流程变量的名称,‘不重要’表示流程变量的值,${}中间的内容要使用boolean类型的表达式,用来判断应该执行的连线。

在这里插入图片描述

十二、排他网关(ExclusiveGateWay)

1,流程图

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

2,部署流程定义+启动流程实例

3,查询我的个人任务

4,完成我的个人任务

/**
	 * 办理任务
	 */
	@Test
	public void completeTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String taskId = "22504";
		// 根据任务ID去完成任务
		taskService.complete(taskId);
		// 根据任务ID去完成任务并指定流程变量
//		taskService.complete(taskId, variables);
		System.out.println("任务完成");
	}
	/**
	 * 办理任务并使用流程变量指定流程走向
	 */
	@Test
	public void completeTask2() {
		TaskService taskService = this.processEngine.getTaskService();
		String taskId = "20004";
		Map<String, Object> variables=new HashMap<>();
		variables.put("money", 1200);
		// 根据任务ID去完成任务并指定流程变量
		taskService.complete(taskId, variables);
		System.out.println("任务完成");
	}

5,说明

1)一个排他网关对应一个以上的顺序流(连线)

2)由排他网关流出的顺序流都有个conditionExpression元素,在内部维护返回boolean类型的决策结果。

3)决策网关只会返回一条结果。当流程执行到排他网关时,流程引擎会自动检索网关出口,从上到下检索如果发现第一条决策结果为true或者没有设置条件的(默认为成立),则流出。

4)如果没有任何一个出口符合条件,则抛出异常

5)使用流程变量,设置连线的条件,并按照连线的条件执行工作流,如果没有条件符合的条件,则以默认的连线离开。

十三、并行网关(parallelGateWay)

1,流程图

在这里插入图片描述

2,部署流程定义+启动流程实例

3,查询我的个人任务

4,完成我的个人任务

/**
	 * 办理任务
	 */
	@Test
	public void completeTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String taskId = "10002";
		// 根据任务ID去完成任务
		taskService.complete(taskId);
		// 根据任务ID去完成任务并指定流程变量
		System.out.println("任务完成");
	}

5,说明

1)一个流程中流程实例只有1个,执行对象有多个

2)并行网关的功能是基于进入和外出的顺序流的
分支(fork):
并行后的所有外出顺序流,为每个顺序流都创建一个并发分支。
汇聚(join):
所有到达并行网关,在此等待的进入分支, 直到所有进入顺序流的分支都到达以后, 流程就会通过汇聚网关。

3)并行网关的进入和外出都是使用相同节点标识

4)如果同一个并行网关有多个进入和多个外出顺序流, 它就同时具有分支和汇聚功能。 这时,网关会先汇聚所有进入的顺序流,然后再切分成多个并行分支。

5)并行网关不会解析条件。 即使顺序流中定义了条件,也会被忽略。

6)并行网关不需要是“平衡的”(比如, 对应并行网关的进入和外出节点数目不一定相等)。如图中标示是合法的:

在这里插入图片描述

十四、个人任务

1,流程图

在这里插入图片描述

2,分配个人任务方式一(直接指定办理人)

在这里插入图片描述

缺点 办理人固定了,但是实际开发中,办理人是不固定的

3,分配个人任务方式二(使用流程变量)

1).部署

2).启动并指定下一个任务的办理人【从session里面取】

	/**
	 * 启动流程并指定下一个任务的办理人
	 */
	@Test
	public void startProcess() {
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
		String processDefinitionKey="HelloWorld";
		Map<String,Object> variables=new HashMap<>();
		variables.put("username", "张三");
		runtimeService.startProcessInstanceByKey(processDefinitionKey,variables);
		System.out.println("流程启动成功");
	}

3). 查询个人任务

4).完成任务并指定下一个任务的办理人

/**
	 * 办理任务
	 */
	@Test
	public void completeTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String taskId="12505";
		Map<String,Object> variables=new HashMap<>();
		variables.put("username", "李四");
		taskService.complete(taskId,variables);
		System.out.println("任务完成");
	}

4,分配个人任务方式三(使用类)

创建一个监听类
作用:当任务到达相应的节点时,会触发监听,让监听器去指定下个任务的办理人

在这里插入图片描述

修改流程图

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

5,总结

个人任务及三种分配方式:
1,在taskProcess.bpmn中直接写 assignee=“张三丰"

2,在taskProcess.bpmn中写 assignee=“#{userID}”,变量的值要是String的,使用流程变量指定办理人

3,使用TaskListener接口,要使类实现该接口,在类中定义,
delegateTask.setAssignee(assignee);// 指定个人任务的办理人
使用任务ID和办理人重新指定办理人:
processEngine.getTaskService()//
.setAssignee(taskId, userId);

十五、用户组任务

1,流程图

在这里插入图片描述

2,分配组任务方式一(直接指定办理人)

1).修改流程图

在这里插入图片描述

2).部署流程

3).相关代码


/**
 * 组任务使用流程变量去设置
 * 
 *
 */
public class GroupTask01 {

	
	/**
	 * 启动流程
	 */
	@Test
	public void startProcess() {
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
		String processDefinitionKey = "HelloWorld";
		runtimeService.startProcessInstanceByKey(processDefinitionKey);
		System.out.println("流程启动成功");
	}

	/**
	 * 查询组任务
	 */
	@Test
	public void findGroupTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String candidateUser = "小B";
		List<Task> list = taskService.createTaskQuery().taskCandidateUser(candidateUser).list();
		if (null != list && list.size() > 0) {
			for (Task task : list) {
				System.out.println("任务ID:" + task.getId());
				System.out.println("流程实例ID:" + task.getProcessInstanceId());
				System.out.println("执行实例ID:" + task.getExecutionId());
				System.out.println("流程定义ID:" + task.getProcessDefinitionId());
				System.out.println("任务名称:" + task.getName());
				System.out.println("任务办理人:" + task.getAssignee());
				System.out.println("################################");
			}
		}
	}

	/**
	 * 查询个人任务
	 */
	@Test
	public void findTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String assignee = "小A";
		List<Task> list = taskService.createTaskQuery().taskAssignee(assignee).list();
		if (null != list && list.size() > 0) {
			for (Task task : list) {
				System.out.println("任务ID:" + task.getId());
				System.out.println("流程实例ID:" + task.getProcessInstanceId());
				System.out.println("执行实例ID:" + task.getExecutionId());
				System.out.println("流程定义ID:" + task.getProcessDefinitionId());
				System.out.println("任务名称:" + task.getName());
				System.out.println("任务办理人:" + task.getAssignee());
				System.out.println("################################");
			}
		}
	}

	/**
	 * 任务拾取
	 */
	@Test
	public void claim() {
		String taskId="2504";
		TaskService taskService = this.processEngine.getTaskService();
		taskService.claim(taskId, "小A");
		System.out.println("任务拾取成功");
	}
	/**
	 * 任务回退
	 */
	@Test
	public void claimBack() {
		String taskId="2504";
		TaskService taskService = this.processEngine.getTaskService();
		taskService.setAssignee(taskId, null);
		System.out.println("任务回退成功");
	}
	
    //5 查询组任务成员列表
    @Test
    public void findGroupUser(){
        String taskId = "2504";
        List<IdentityLink> list = processEngine.getTaskService()//
                        .getIdentityLinksForTask(taskId);
         //List<IdentityLink> list = processEngine.getRuntimeService()//
        //                .getIdentityLinksForProcessInstance(instanceId);
        for(IdentityLink identityLink:list ){
            System.out.println("userId="+identityLink.getUserId());
            System.out.println("taskId="+identityLink.getTaskId());
            System.out.println("piId="+identityLink.getProcessInstanceId());
            System.out.println("TYPE="+identityLink.getType());
            System.out.println("######################");
        }
    }
	

	/**
	 * 办理任务
	 */
	@Test
	public void completeTask() {
		TaskService taskService = this.processEngine.getTaskService();
		String taskId = "5002";
		taskService.complete(taskId);
		System.out.println("任务完成");
	}
}


3,分配组任务方式二(使用流程变量)

1).修改流程图

在这里插入图片描述

2) 部署流程

3).相关代码

/**
 * 组任务使用流程变量去设置
 * 
 * @author LJH
 *
 */
public class GroupTask01 {

	/**
	 * 启动流程
	 */
	 @Test
	public void startProcess() {
		RuntimeService runtimeService = this.processEngine.getRuntimeService();
		String processDefinitionKey = "HelloWorld";
		Map<String,Object> variables=new HashMap<>();
		variables.put("usernames", "小A,小B,小C,小D");
		runtimeService.startProcessInstanceByKey(processDefinitionKey,variables);
		System.out.println("流程启动成功");
	}

(其他与上面一致)

4,分配组任务方式三(使用类)

1).创建监听器

在这里插入图片描述

2).修改流程图

在这里插入图片描述

3).相关代码

与第一种方法一致

5,总结

在这里插入图片描述

1)组任务及三种分配方式,

1、在taskProcess.bpmn中直接写 candidate-users=“小A,小B,小C,小D"

2、在taskProcess.bpmn中写 candidate-users =“#{userIDs}”,变量的值要是String的。

使用流程变量指定办理人

	Map<String, Object> variables = new HashMap<String, Object>();
	variables.put("userIDs", "大大,小小,中中");

3、使用TaskListener接口,使用类实现该接口

	delegateTask.addCandidateUser(userId1);//添加组任务的用户
	delegateTask.addCandidateUser(userId2);
	//组任务分配给个人任务(认领任务),
     processEngine.getTaskService().claim(taskId, userId);
	//个人任务分配给组任务,
     processEngine.getTaskService(). setAssignee(taskId, null);
	//向组任务添加人员,
     processEngine.getTaskService().addCandidateUser(taskId, userId);
	//向组任务删除人员,
     processEngine.getTaskService().deleteCandidateUser(taskId, userId);

2)个人任务和组任务存放办理人对应的表,

act_ru_identitylink表存放任务的办理人,包括个人任务和组任务,表示正在执行的任务
act_hi_identitylink表存放任务的办理人,包括个人任务和组任务,表示历史任务

区别在于,如果是个人任务TYPE的类型表示participant(参与者)
如果是组任务TYPE的类型表示candidate(候选者)和participant(参与者)

十六、Activiti总结

在这里插入图片描述

在这里插入图片描述

  • 5
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值