Activiti5

Activiti5

概述
    工作流(workflow)
    为了完成一项任务或事情,需要多个参与者同时或流水式共同完成。
    目前应用比较广泛的行业:物流,办公管理系统
    是一个半成品项目,需要做二次开发
解决的问题
    为了实现某个业务目标,利用计算机在多个参与者之间按某种预定规则自动传递文档、信息或者任务,从而提高业务目标效率。
发展史
    jbpm(jboss公司)
    BPM:Business Process Management 业务流程管理
    BPMN:Business Process Modeling and Notation 业务流程建模与标注规范
下载
    官网:http://www.activiti.org
文件夹说明
    database
        操作数据库sql脚本
    docs
        api文档与用户指南
    xsd
        xml约束文件
    libs
        jar包
    wars
        web应用
安装
    流程设计器离线安装(整合到eclipse中)
        把activiti-plugin拷贝到eclipse\dropins中
    流程设计器在线安装
        eclipse --> help --> install new software
        Activiti BPMN 2.0 Designer
        http://activiti.org/designer/update
    需要创建数据库表(24张)
        activiti.mysql55.create.engine.sql
        activiti.mysql55.create.history.sql
        activiti.mysql.create.identity.sql
        直接运行sql文件中的sql即可。
24张表介绍
    ACT_RE_*(3张表)
        仓储数据
        act_re_deployment
            部署的流程
        act_re_procdef
            流程定义
    ACT_RU_*
        运行过程中的流程数据
        act_ru_variable
            运行中的变量存储位置
    ACT_ID_*
        权限管理数据
    ACT_HI_*
        历史的数据
    ACT_GE_*
        公共数据
        act_ge_bytearray 
            部署的流程图
    ACT_EVT_LOG
        事件日志
项目中引入jar包
    第一种方式
        拷贝jar包
    第二种方式
        maven方式
核心类
    ProcessEngineConfiguration
        流程引擎配置信息对象
    ProcessEngine
        流程引擎
    RepositoryService 
        仓储服务
    RuntimeService
        运行时服务
    TaskService
        任务服务
    IdentityService
        身份管理服务
    HistoryService 
        历史服务
    FormService
        表单数据服务
    ManagementService
        管理服务
ProcessEngineConfiguration(流程引擎配置信息对象)创建方式
    第一种
        createProcessEngineConfigurationFromResourceDefault()
            需要加载src/activiti.cfg.xml
            bean的id名称一定为:processEngineConfiguration
    第二种
        createProcessEngineConfigurationFromResource(String resource)
            需要加载src/activiti.cfg.xml(配置文件可以改名)
            bean的id名称一定为:processEngineConfiguration
    第三种
        createProcessEngineConfigurationFromResource(String resource, String beanName)
            需要加载src/activiti.cfg.xml(配置文件可以改名)
            bean的id名:processEngineConfiguration(可以自己给名字)
    第四种
        createProcessEngineConfigurationFromInputStream(InputStream inputStream)
            需要加载src/activiti.cfg.xml(配置文件可以改名)
            bean的id名称一定为:processEngineConfiguration
    第五种
        createProcessEngineConfigurationFromInputStream(InputStream inputStream, String beanName)
            需要加载src/activiti.cfg.xml(配置文件可以改名)
            bean的id名:processEngineConfiguration(可以自己给名字)
    第六种
        createStandaloneProcessEngineConfiguration().
说明
    流程:指完成某件事情需要的过程.
    流程定义:具体的流程.把具体流程定义在bpmn文件中.
    流程部署:把流程定义文件保存到数据库.
Activiti5:工作流
1. RepositoryService仓储服务:
   -- act_ge_bytearray
   -- act_re_deployment
   -- act_re_procdef
   -- act_ge_property

   完成的功能:
   -- 流程部署
   -- 流程部署查询
   -- 删除流程部署
   -- 流程定义查询
   -- 查询流程定义资源文件(BpmnModel)
   -- 查询流程定义资源图片(InputStream)

2. RuntimeService运行时服务:
   -- act_ru_execution
   -- act_ru_event_subscr
   -- act_ru_job
   -- act_ru_task
   -- act_ru_variable

   完成的功能:
   -- 开启流程实例.
   -- 查询流程实例.
   -- 删除流程实例.
   -- 触发走一步.signal
   -- 获取当前活动的节点.

3. TaskService任务服务:
   -- act_ru_task
   完成的功能:
   -- 查询任务.
   -- 完成任务.
  
4. HistoryService历史服务:
   -- act_hi_procinst
   -- act_hi_taskinst

   完成的功能:
   -- 查询历史流程实例
   -- 查询历史任务实例

新增流程部署

使用
    RepositoryService服务
    获取方式:
        ProcessEngine.getRepositoryService()获取仓储服务对象
    步骤
        获取部署构建对象
            DeploymentBuilder deploymentBuilder = RepositoryService.createDeployment()
        添加流程定义资源文件
            deploymentBuilder.addInputStream(String resourceName, InputStream inputStream) 
            addZipInputStream(ZipInputStream zipInputStream)
        部署流程定义
            deploymentBuilder.deploy();
刚刚部署的流程在这三张表中有数据
    act_re_deployment 部署表
        NAME_ 文件名字
        CATEGORY_ 添加种类
        DEPLOY_TIME_ 部署时间
    act_ge_bytearray
        
    act_re_procdef
        myProcess_1:3:2508
        act_re_procdef表的KEY_:act_re_procdef表的VERSION_:act_ge_bytearray表的自增长id最大值+1
    

查询所有部署的流程(流程部署)

使用
    RepositoryService服务
    获取方式:
        ProcessEngine.getRepositoryService()获取仓储服务对象
    查询方式:
        查询对象:
            DeploymentQuery deploymentQuery = repositoryService.createDeploymentQuery();
        api:
            orderByDeploymenTime()
                根据部署时间排序
            orderByDeploymentId()
                根据部署id排序
            orderByDeploymentName()
                根据部署名称排序
            orderByTenantId()
                
            desc()
                降序
            asc()
                升序
            list()
                全部查询
        说明
            List<Deployment> deploymentLists = repositoryService.createDeploymentQuery().orderByDeploymenTime().desc().list();
详细说明
    编号 d.id
    名称 d.name
    种类 d.category
    部署时间 d.deploymentTime

删除部署的流程(流程部署)

使用
    RepositoryService服务
    获取方式:
        ProcessEngine.getRepositoryService()获取仓储服务对象
    删除方式:
        RepositoryService.deleteDeployment(编号, true);
            编号是 act_re_deployment表中的ID_ 
            repositoryService.deleteDeployment(deploymentId, true);

查看所有的流程定义

是部署的所有流程的详细信息,根据流程定义来启动流程
使用
    RepositoryService服务
    获取方式:
        ProcessEngine.getRepositoryService()获取仓储服务对象
    api
        查询对象:
            ProcessDefinitionQuery processDefinitionQuery = repositoryService.createProcessDefinitionQuery();
        latestVersion()
            每个流程定义的最后版本(最新版本)
        orderByProcessDefinitionVersion()
            根据流程定义的版本号排序
        desc()
            降序
        list()
            全部查询
        说明
            List<ProcessDefinition> processDefinitionLists = repositoryService.createProcessDefinitionQuery().latestVersion().orderByProcessDefinitionVersion().desc().list();
详细说明
    流程定义ID  p.id
    名称  p.name
    部署ID  p.deploymentId
    key  p.key
    资源文件名称  p.resourceName
    资源图片名称  p.diagramResourceName
    版本号  p.version

根据流程定义开启流程实例

概述
    比如开启一个请假流程
使用
    根据流程定义的ID开启
        RuntimeService runtimeService = pe.getRuntimeService();
        runtimeService.startProcessInstanceById(pdId);
        注意:
            是根据act_re_procdef表中的ID_开启
    根据流程定义的Key开启
        RuntimeService runtimeService = pe.getRuntimeService();
        runtimeService.startProcessInstanceByKey(pdKey);
        注意:
            是根据act_re_procdef表中的KEY_开启
开启一个流程实例表中数据说明
    act_ru_execution 
        流程实例表
        PROC_INST_ID_
            流程定义act_re_procdef表的id
        ACT_ID_
            当前接收要处理任务节点
    act_hi_actinst
        ??
    act_hi_procinst
        ??

查看某个流程定义的所有流程实例

概述
    比如查询技术部所有正在请假的流程
使用
    查询方式
        RuntimeService runtimeService = pe.getRuntimeService();
        ProcessInstanceQuery piq = runtimeService.createProcessInstanceQuery();
        List<ProcessInstance> piLists = piq.processDefinitionId(pdId).orderByProcessInstanceId().asc().list();
    说明
        根据act_re_procdef表的ID_来查询
详细说明
    流程实例ID  p.id
    流程实例ID  p.processInstanceId
    名称  p.processDefinitionName
    流程定义ID  p.processDefinitionId
    流程定义版本号  p.processDefinitionVersion
    活动的节点ID  p.activityId

删除某个流程定义的某个流程实例

概述
    例如把老王的请假流程删除
使用
    删除方式
        RuntimeService runtimeService = pe.getRuntimeService();
        runtimeService.deleteProcessInstance(piId, "删除");
    说明
        根据act_ru_execution表的ID_去删除
        删除后在act_hi_procinst表中可以看到删除的原因

查看某个流程定义的某个流程实例的流程图

概述
    例如把老王的请假流程走到哪一步了.
使用
    查看方式
        RepositoryService repositoryService = pe.getRepositoryService();
        RuntimeService runtimeService = pe.getRuntimeService();
        ProcessInstance pi = runtimeService.createProcessInstanceQuery().processInstanceId(piId).singleResult();
        InputStream inputStream = repositoryService.getProcessDiagram(pi.getProcessDefinitionId());
        BufferedImage image = ImageIO.read(inputStream);
        Graphics2D graphics = (Graphics2D)image.getGraphics();
        graphics.setColor(Color.RED);
        graphics.setStroke(new BasicStroke(3.0f));
        graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
        List<String> activityIds = runtimeService.getActiveActivityIds(piId);
        BpmnModel bpmnModel = repositoryService.getBpmnModel(pi.getProcessDefinitionId());
        for (String activityId : activityIds){
            GraphicInfo gi = bpmnModel.getGraphicInfo(activityId);
            graphics.drawRoundRect((int)gi.getX(), (int)gi.getY(),(int)gi.getWidth(), (int)gi.getHeight(), 10, 10);
        }
        response.setContentType("images/png");
        ImageIO.write(image, "png", response.getOutputStream());
    说明
        根据act_ru_execution表的ID_去查询  

触发流程实例进行下一步

概述
    例如:领导同意老王请假,请下一个领导批准
使用
    触发方式
        RuntimeService runtimeService = pe.getRuntimeService();
        runtimeService.signal(piId);
    说明
        根据根据act_ru_execution表的ID_去触发

画流程图

说明
    BPMN的6大部分
        Event事件
            Start Event开始事件
            End Event结束事件
            Boundary Event边界事件
            Intermediate Event中间捕获事件
        Tasks任务
            UserTask用户任务(人机交互)
        Gateways网关
            ParallelGateway并行网关(多人必须同时审批)
        Container容器
        Connection连接
        Artifacts说明
Task任务节点
    UserTask用户任务
        General
            id 唯一标志
            name 起一个名称
        Main Config
            Assignee 任务处理人  
                admin 这种写法是指定具体的用户
                ${user} 这种写法是给一个流程变量
            Candidate users(comma separated) 任务候选人(可以写很多,用逗号隔开)
            Candidate group(comma separated) 任务候选人组
            Form key 表单的key - 用的少 
            Due date(Variable) 过期时间 - 用的少
                P:代表一段时间(年月日)Y年M月D日
                T:代表后面为(时分秒)H时M分S秒
                例如PT10H10M10S(代表10小时10分10秒)
            priority 优先级
            Category 种类,说明
            Skip expression 跳过模式(当达到条件就跳过去)
        Documentation
            Documentation 任务的描述信息
        Form 
            表单
        Listeners
            监听器
        Multi instance
            多实例
            Sequential 是否有顺序 false表示无序,true表示有序.
    在act_ru_task表中会有每个人的任务
Connection连接线
    SequenceFlow 任务连接线

查看自己任务

说明
    用户要登录后查看自己要处理的任务
    例如:领导要看自己待处理的任务
使用
    查询方式
        TaskService taskService = pe.getTaskService();
        TaskQuery taskQuery = taskService.createTaskQuery();
        HttpSession session = request.getSession();
        taskQuery.taskAssignee(userid);//用户的id
        List<Task> tasklist = taskQuery.list();
    注意:
        在画流程图中在任务节点上写明这点.(Main Config的Assignee)
详细说明
    任务ID  t.id
    任务名称  t.name
    任务处理人  t.assignee
    任务创建时间  t.createTime
    流程实例ID  t.processInstanceId

处理自己的任务

说明
    用户必须要先领取任务,才能处理
使用
    执行任务
        TaskService taskService = pe.getTaskService();
        taskService.complete(taskId);//任务的id
    说明
        是act_ru_task表中的ID_

根据流程实例id查询流程实例

说明
    管理员处理其他用户的流程
使用
    查询id查询
        TaskService taskService = pe.getTaskService();
        TaskQuery tq = taskService.createTaskQuery();
        List<Task> taskLists = tq.processInstanceId(piId).list();
详细说明
    任务ID  t.id
    任务名称  t.name
    任务处理人  t.assignee
    任务创建时间  t.createTime
    流程实例ID  t.processInstanceId

查看历史流程实例

说明
    查看所有提过哪些流程
使用
    HistoryService historyService = pe.getHistoryService();
    HistoricProcessInstanceQuery hpiq = historyService.createHistoricProcessInstanceQuery();
    List<HistoricProcessInstance> hpiLists = hpiq.orderByProcessInstanceId().asc().list();
详细说明
    历史流程实例ID p.id
    开始时间 p.startTime
    结束时间 p.endTime
    完成的总时间 p.durationInMillis
    开启流程实例用户 p.startUserId
    删除原因 p.deleteReason

查看历史流程实例中的所有任务

说明
    查看某个流程实例中的所有任务
使用
    HistoryService historyService = pe.getHistoryService();
    HistoricTaskInstanceQuery htiq = historyService.createHistoricTaskInstanceQuery();
    List<HistoricTaskInstance> htiLists = htiq.processInstanceId(hpiId).orderByHistoricTaskInstanceEndTime().asc().list();
详细
    任务ID   p.id
    任务名称  p.name
    任务处理人  p.assignee
    任务开始时间 p.startTime
    任务结束时间  p.endTime
    任务完成的时间 p.durationInMillis
    任务分派时间 p.claimTime
    任务工作总时间 p.workTimeInMillis

查看自己的任务(任务候选人中有自己)

概述
    请假由三个领导谁批都可以,那么谁先登录谁批准.
    拉模式
使用
    TaskService taskService = pe.getTaskService();
    TaskQuery tq = taskService.createTaskQuery();
    List<Task> candidateUserTaskLists = tq.taskCandidateUser(userId).list();//根据候选人查询
    for (Task task : candidateUserTaskLists){
        taskService.claim(task.getId(), userId);//领取任务
    }
    List<Task> taskLists = taskService.createTaskQuery().taskAssignee(userId).list();
详细说明
    任务ID  t.id
    任务名称  t.name
    任务处理人  t.assignee
    任务创建时间  t.createTime
    流程实例ID  t.processInstanceId

查看自己的任务(任务组中有自己)

概述
    请假由三个领导谁批都可以,那么谁先登录谁批准.
    拉模式
使用
    Map<String, List<String>> maps = new HashMap<>();
    List<String> users = new ArrayList<>();
    users.add("test1");
    users.add("test2");
    maps.put("role1", users);
    for (Map.Entry<String, List<String>> entry : maps.entrySet()){
        if (entry.getValue().contains(userId)){
            List<Task> candidateGroupTaskLists = tq.taskCandidateGroup(entry.getKey()).list();
            for (Task task : candidateGroupTaskLists){
                taskService.claim(task.getId(), userId);// 领取任务
            }
        }
    }
    List<Task> taskLists = taskService.createTaskQuery().taskAssignee(userId).list();

启动流程实例设置变量

概述
    在任务启动或者流转过程中会添加一些变量,这些变量用于辨别不同的用户处理流程,或者记录一些信息
使用
    RuntimeService runtimeService = pe.getRuntimeService();
    Map<String, Object> params = new HashMap<>();
    params.put("user", "lee2");
    params.put("age", 18);
    params.put("u", new User());
    runtimeService.startProcessInstanceById(pdId, params);

任务监听器

概述
    UserTask 中设置Listeners中的Task listeners.
设置说明
    Task Listener 有四种事件
        create 创建的时候(任务在创建的时候)
        assignment 分配用户的时候
        complete 完成任务的时候
        all 全部情况
        使用
            Java类的方式 
                Task Listener监听器类必须要实现TaskListener接口
                public class CreateTaskListener implements TaskListener {}
                public class AssignmentTaskListener implements TaskListener {}
                public class CompleteTaskListener implements TaskListener {}
            Expression表达式的方式
                #{user.test()}  这个是在调用user对象的test方法。这个user是流程变量。#{user.test(100,'定州')}
    Execution listener 执行监听器
        start 任务开始
        end 任务结束
        使用
            Java类的方式
                Execution listener 监听器必须要实现ExecutionListener接口
                public class StartListener implements ExecutionListener {}
                public class EndListener implements ExecutionListener {}
使用
    在监听器中获取实例中的变量
    public class CreateTaskListener implements TaskListener {
        private Expression name;
        public void notify(DelegateTask task) {
            //表达式的文本 name.getExpressionText()
            //name的值 name.getValue(task)
            //System.out.println(task.getVariables());// 获取所有的流程变量
            //System.out.println("CreateTaskListener事件名:" + task.getEventName());// create
            String userId = (String)task.getVariable("user");// 获取指定的流程变量
            task.setAssignee(userId);// 分配任务处理人
            //还可以添加任务候选人(task.addCandidateUser),用户组(task.addCandidateGroup).用于动态的分配任务
        }
    }
    public class AssignmentTaskListener implements TaskListener {
        public void notify(DelegateTask task) {
            System.out.println(task.getVariables());// 获取所有的流程变量
            System.out.println(task.getAssignee());// 获取任务处理人
            System.out.println("AssignmentTaskListener事件名:" + task.getEventName());// assignment
        }
    }
    public class CompleteTaskListener implements TaskListener {
        public void notify(DelegateTask task) {
            System.out.println(task.getVariables());// 获取所有的流程变量
            System.out.println("CompleteTaskListener事件名:" + task.getEventName());// create
            System.out.println(task.getAssignee());// 任务处理人
        }
    }
    public class StartListener implements ExecutionListener {
        private Expression id;
        public void notify(DelegateExecution execution) throws Exception {
            System.out.println(id.getValue(execution));
            System.out.println(execution.getVariables());
            System.out.println("StartListener事件名" + execution.getEventName());
            execution.setVariable("admin", "aaaa");//设置 流程变量
        }
    }

表单数据服务

说明
    在任务中设置表单字段与数据,在监听器中获取表单数据
使用
    获取表单数据
        public class CreateTaskListener implements TaskListener {
            public void notify(DelegateTask task) {
                FormService fs = task.getExecution().getEngineServices().getFormService();
                List<FormProperty> lists = fs.getTaskFormData(task.getId()).getFormProperties();
                for (FormProperty fp : lists){
                    System.out.println(fp.getId() + "==" + fp.getName() + "==" + fp.getValue());
                }
            }
        }
    启动流程实例时设置表单数据
        RuntimeService runtimeService = pe.getRuntimeService();
        Map<String, Object> params = new HashMap<>();
        params.put("name", "lee2");
        params.put("sex", 1);
        untimeService.startProcessInstanceById(pdId, params);
定义表单属性
    id 字段id
    name 字段名称
    type 字段类型
    expression 字段表达式${name}
    variable 变量名
    default 默认值
    date pattern 日期格式化
    readable 可读
    writeable 可写
    required 必须要传
    

多实例

multi instance
    sequential true代表有序,false代表无序
        无序:表示有多个用户,一次创建全部任务显示.用的比较多.
        true有序: 需要一个一个的签到,而且有序.一个都不能少
    loop cardinality  循环次数
    collection 用户的集合,要放到流程变量中.例如是userList
    element variable 变量,会放到流程变量中.类似于遍历时的别名
    completion condition 完成条件, ${nrOfCompletedInstances / nrOfInstances >= 0.6 }
        nrOfCompletedInstances是已经完成的个数
        nrOfInstances是总个数
使用
    启动此类流程定义,一定要在启动的时候,给一个userList的流程变量
        RuntimeService runtimeService = pe.getRuntimeService();
        Map<String, Object> params = new HashMap<>();
        List<String> userLists = new ArrayList<>();
        userLists.add("admin1");
        userLists.add("admin2");
        params.put("userLists", userLists);
        runtimeService.startProcessInstanceById(pdId, params);
有序时
    需要四个条件都要给
无序时
    循环次数不需要给了
重点动态分配用户集合:
    添加create监听器
    public class CreateTaskListener implements TaskListener {
        private String[] users = {"test1", "test2", "test3", "test4", "test5"};//定义用户数组
        public void notify(DelegateTask task) {
            int i = task.getVariable("loopCounter", Integer.class);// 获取循环计数器
            task.setAssignee(users[i]);// 分配任务处理人
        }
    }

脚本任务(自动触发)

概述
    ScriptTask
    自己触发,自己完成
    写JavaScript代码
说明
    General
        name 任务名称
    MainConfig
        可以写两种语言,我们写js代码
        测试代码
            var sum = 0;
            for (var i = 0; i <= 100; i++){
                sum += i;
            }
            execution.setVariable("sum", sum);//设置变量
        

服务任务(自动触发)

概述
    ServiceTask
    自己触发,自己完成
    配置java类进行后台任务处理
说明
    General
        name 任务名称
    MainConfig
        必须要指定一个服务的java类,这个java类必须实现JavaDelegate接口.
        可以在Documentation中设置变量,这个变量就可以在服务类中获取到.
        表达式类型就是指定调用某个对象中的方法
    测试
        public class ServiceTaskDelegate implements JavaDelegate {}

手工任务(自动触发)

概述
    ManualTask
    自己触发,自己完成
说明
    General
        name 任务名称

接收任务(自动触发)

概述
    ReceiveTask
    自己触发,自己完成
说明
    General
        name 任务名称
注意:
    需要用信号的方式触发.

邮件任务(自动触发)

概述
    MailTask
    自己触发,自己完成
    要作用是用来发邮件
jar包
    依赖jar: (mail.jar、activation.jar、common-mail.xx.jar)
说明
    General
        name 任务名称
    MainConfig
        to 邮件发送给谁 ${to}
        from 谁发送的邮件 ${from}
        subject 邮件主题 ${subject}
        Cc 抄送给谁 ${to}
        Bcc 秘抄送 ${to}
        Charset 邮件编码 utf-8
        html html邮件 ${to},你的订单已审核!
        non html 文本邮件 ${to},你的订单已审核!
使用
    在创建流程引擎配置(ProcessEngineConfiguration)的时候设置用户名和密码
        ProcessEngineConfiguration pec = ProcessEngineConfiguration.createStandaloneProcessEngineConfiguration();
        pec.setDatabaseSchemaUpdate(ProcessEngineConfiguration.DB_SCHEMA_UPDATE_TRUE)
       .setDatabaseType("mysql") // 数据库的类型
       .setJdbcDriver("com.mysql.jdbc.Driver") // 连接数据库驱动
       .setJdbcUrl("jdbc:mysql://localhost:3306/activiti") // 连接数据库的URL
       .setJdbcUsername("root") // 连接数据库的用户名
       .setJdbcPassword("root") // 连接数据库的密码
       .setActivityFontName("宋体") // 设置活动节点上的字体
       .setLabelFontName("宋体")// 设置连接线上的字体
       .setXmlEncoding("utf-8") // 设置流程定义文件的编码
       .setMailServerDefaultFrom("476570365@qq.com") // 设置发送人
       .setMailServerHost("smtp.qq.com") // 设置邮件服务器 
       .setMailServerUsername("476570365@qq.com") // 设置用户名
       .setMailServerPassword("xxxxxx"); // 设置密码
    启动流程定义的时候,设置发给谁
        RuntimeService runtimeService = pe.getRuntimeService();
        Map<String, Object> params = new HashMap<>();
        params.put("to", "476570365@qq.com");
        params.put("from", "476570365@qq.com");
        params.put("subject", "订单审批信息");
        runtimeService.startProcessInstanceById(pdId, params);

开始事件

说明
    Start Event开始事件
分类
    StartEvent正常开始事件
    TimerStartEvent定时开始事件
        不需要用户启动
    MessageStartEvent消息开始事件
    SignalStartEvent信号开始事件
StartEvent正常开始事件
    General
        name 名称
    MainConfig
        initiator 事件是由谁来启动
    Form
        表单数据
    里面的流程变量给的时候要在启动的时候给
        RuntimeService runtimeService = pe.getRuntimeService();
        pe.getIdentityService().setAuthenticatedUserId("test888");//设置鉴权用户(开启流程实例用户)
        List<FormProperty> lists =pe.getFormService().getStartFormData(pdId).getFormProperties();// 获取开始事件中的表单数据
        for (FormProperty fp : lists){
            System.out.println(fp.getId() + "==" + fp.getName() + "==" + fp.getValue());
        }
        runtimeService.startProcessInstanceById(pdId);
TimerStartEvent定时开始事件
    General
        name 名称
    MainConfig 启动时间,写一个就可以
        time duration 程序启动后延迟多长时间启动
            PT10H10M10S(代表10小时10分10秒)
        time date 指定一个时间
            2017-11-5T10:05:05
        time cycle 可以循环多少次,可以开启多个流程实例
            R4/PT10S 每个10秒启动一次
            R4/2016-01-23T17:07:00/PT10S 在一个时间后延迟10秒
            R4代表启动4次,
    使用
        需要在创建流程引擎配置(ProcessEngineConfiguration)的时候设置开启
        ProcessEngineConfiguration pec = ProcessEngineConfiguration.createStandaloneProcessEngineConfiguration();
        pec.setJobExecutorActivate(true); // 设置激活任务调度
MessageStartEvent消息开始事件
    General
        name 名称
    MainConfig
        Message ref 引用一个流程定义已经设置好的消息
        特别注意:需要自己去写消息的引用
            <startEvent id="messagestartevent1" name="Message start">
                <messageEventDefinition messageRef="myMsg"></messageEventDefinition>
            </startEvent>
    注意:
        一定要指定是一个什么样的消息.
        在流程定义的图中设置消息类型.
            Messages
                Message definitions 中定义消息的id和name名称
    使用
        启动方式
            RuntimeService runtimeService = pe.getRuntimeService();
            runtimeService.startProcessInstanceByMessage("myMsg2");//根据消息id来启动
SignalStartEvent信号开始事件
    General
        name 名称
    MainConfig
        Signal ref 引用一个流程定义已经设置好的信号
    注意:
        一定要指定全局的信号
        在流程定义的图中设置消息类型.
            Signals
                Signal definitions 中定义信号的id和name名称  
    使用
        启动方式
            RuntimeService runtimeService = pe.getRuntimeService();
            runtimeService.signalEventReceived("mySignal");// 信号事件接收启动流程实例

结束事件

说明
    End Event结束事件
分类
    EndEvent结束事件
    TerminateEndEvent终止结束事件
    ErrorEndEvent错误结束事件(会引发错误码)子流程
EndEvent结束事件
    自动就会结束,正常结束
TerminateEndEvent终止结束事件
    标记一下不是正常结束,是被终止的
ErrorEndEvent错误结束事件(会引发错误码)子流程
    在子流程中使用

边界事件

说明
    Boundary Event边界事件
分类
    TimerBoundaryEvent定时边界事件
    MessageBoundaryEvent消息边界事件
    SignalBoundaryEvent信号边界事件
    ErrorBoundaryEvent错误边界事件(接收错误码)子流程
TimerBoundaryEvent定时边界事件
    把这个事件放到某个任务上即可.
    例如:用户不收货,就定时7日后自动收货结束流程。
    MainConfig
        Cancle activity 
        time duration 
        time date 指定一个时间
        time cycle 可以循环多少次
MessageBoundaryEvent消息边界事件
    把这个事件放到某个任务上即可.
    注意:
        需要定义一个全局的消息.
        然后在消息事件中引用这个消息.MainConfig(Message ref)
            <boundaryEvent id="boundarymessage1" name="Message" attachedToRef="usertask3" cancelActivity="true">
                <messageEventDefinition messageRef="myMsg"></messageEventDefinition>
            </boundaryEvent>
    使用
        RuntimeService runtimeService = pe.getRuntimeService();
        Execution execution = runtimeService.createExecutionQuery().messageEventSubscriptionName("myMsg").singleResult();//根据消息名查询执行事件id
        runtimeService.messageEventReceived("myMsg", execution.getId());//消息事件接收完成当前边界消息事件
SignalBoundaryEvent信号边界事件
    把这个事件放到某个任务上即可.
    注意:
        需要定义一个全局的信号.
    使用
        RuntimeService runtimeService = pe.getRuntimeService();
        Execution execution = runtimeService.createExecutionQuery().signalEventSubscriptionName("mySignal").singleResult();//根据信号名查询执行对象
        runtimeService.signalEventReceived("mySignal", execution.getId());//信号事件接收完成当前信号边界事件
ErrorBoundaryEvent错误边界事件(接收错误码)子流程
    把这个事件放到子流程中使用

中间捕获事件

说明
    Intermediate Event中间捕获事件
分类
    TimerCatchingEvent定时捕获事件
    MessageCatchingEvent消息捕获事件
    SignalCatchingEvent信号捕获事件
    NoneThrowingEvent空引发事件
TimerCatchingEvent定时捕获事件
    定时触发,进入下一环节.
    MainConfig
        time duration 延迟多少秒
        time date 指定一个时间
        time cycle 可以循环多少次
MessageCatchingEvent消息捕获事件
    定义一个全局消息
    使用
        RuntimeService runtimeService = pe.getRuntimeService();
        Execution execution = runtimeService.createExecutionQuery().messageEventSubscriptionName("myMsg").singleResult();
        runtimeService.messageEventReceived("myMsg", execution.getId());
SignalCatchingEvent信号捕获事件
    定义一个全局信号
    使用
        RuntimeService runtimeService = pe.getRuntimeService();
        Execution execution = runtimeService.createExecutionQuery().signalEventSubscriptionName("mySignal").singleResult();
        runtimeService.signalEventReceived("mySignal", execution.getId());
NoneThrowingEvent空引发事件
    自动触发,直接进入下一个流程

网关

概述
    比较常用
分类
    ParallelGateway并行网关(多人必须同时审批)
    ExclusiveGateway排他网关(多人只能选择一人审批)
    InclusiveGateway包含网关(集成了并行与排他网关)
    EventGateway事件网关(只能连接中间捕获事件)
ParallelGateway并行网关(多人必须同时审批)
    注意
        多个人任务之前和之后都要有一个并行网关.
        不需要加条件
    说明
        也就是两个人必须同时完成任务,流程才能继续进行
ExclusiveGateway排他网关(多人只能选择一人审批)
    例如:请假1天,请假两天,请假三天以上等等的流程情况
    注意
        需要在流程线上加条件
        MainConfig(Condition  ${leaveDay <= 2})
    使用:
        启动时需要加上流程变量
        RuntimeService runtimeService = pe.getRuntimeService();
        Map<String, Object> params = new HashMap<String, Object>();
        params.put("leaveDay", 4);
        runtimeService.startProcessInstanceById(pdId, params);
InclusiveGateway包含网关(集成了并行与排他网关)
    在某条线上加条件,就是排他网管,不加条件的就是并行网关
EventGateway事件网关(只能连接中间捕获事件)
    只能连接边界事件

容器

说明
    Container容器
内容
    SubProcess子流程
    Call Activity调用活动(子流程)
    Pool池 | Lane小巷
SubProcess子流程
    例如
        过年买票
        1.发起流程
        2.买票
        3.子流程(交给黄牛去买票)SubProcess
            3.1发起流程
            3.2黄牛买票
            3.3排他网管
            3.4买到票正常结束EndEvent
            3.5没买到票错误结束事件ErrorEndEvent
        4.子流程的错误边界事件ErrorBoundaryEvent
            4.1做汽车回家
        5.子流程正常结束
            5.1坐火车回家
    这两个节点必须要在子流程中
        ErrorEndEvent错误结束事件(会引发错误码)子流程
        ErrorBoundaryEvent错误边界事件(接收错误码)子流程
Call Activity调用活动(子流程)
    Task任务中有一个CallActivity调用活动节点
    使用
        1.准备两个流程文件
        2.配置CallActivity
            MainConfig
                Called element 设置另外一个流程文件的id
        3.都部署就可以了。
Pool池和Lane小巷
    Lane小巷要放在Pool池中。
    可以让人看得比较清晰
    例如:请假流程
        技术部小巷
            1.发起流程
            2.主管审批
            3.经理审批
        总经办
            4.总经理审批
        人事部
            5.审批
            6.结束流程

ZIP部署

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,988评论 25 707
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,941评论 6 13
  • 静――白天总是很吵,有人抱怨他的热闹,你说你什么都不想说,相信我能够自己明了。可是,我是一把小提琴啊,在你这深夜,...
    噢呢呢呢阅读 546评论 0 3
  • S7世界总决赛马上就要开始打了,而这次的S赛呢!可能有点不同,有一支日本队也将参加那个什么名额赛吧!他们也将会来我...
    黄铜刀阅读 364评论 0 0