Activiti工作流引擎的使用、思考与总结(下)

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: Activiti工作流引擎的使用、思考与总结
  • 流程实例
// 考虑到每一个流程都有一个开始节点,那么我就抓住它们的共同点,做了一个抽象类,供所有的实例类来继承
public abstract class BaseProcessServiceImpl extends AbstractElement {
    private ProcessDefinition processDefinition;
    private BpmnModel bpmnModel;
    private StartEvent startEvent;
    private Collection<FlowElement> xmlNodeElements;
    private static final String START_EVENT_KEY = "0";
    // 通过IOC容器获取所有的用户任务处理器对象;后续会提到UserTaskHandler
    @Autowired private Map<String, UserTaskHandler> userTaskHandlers;
    // 持久化工具;专门用来存储数据
    @Autowired private ActivitiBusinessDataRepository activitiBusinessDataRepository;
    // 初始化工作;这些解析出来的属性都是为了方便以后扩展使用。
    @PostConstruct
    private void init() {
        // 把流程定义解析出来
        processDefinition =
                repositoryService
                        .createProcessDefinitionQuery()
                        .processDefinitionKey(processDefinitionKey())
                        .latestVersion()
                        .singleResult();
        // 解析出bpmn模型
        bpmnModel = repositoryService.getBpmnModel(processDefinition.getId());
        Process process = bpmnModel.getProcessById(processDefinitionKey());
        // 解析出所有的xml节点元素
        this.xmlNodeElements = process.getFlowElements();
        // 解析出启动节点
        this.startEvent = (StartEvent) process.getInitialFlowElement();
    }
    /**
     * 启动流程实例
     *
     * @param businessKey
     * @param executor 执行人
     * @param variables 需要放进流程实例中的变量值
     * @param data 用来贯穿整个方法,可以把数据放在里面,最后在调用的方法中取出来使用
     * @return
     * @throws GlobalException
     */
    @Transactional(rollbackFor = Exception.class)
    public ProcessInstance startProcessInstance(
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data)
            throws GlobalException {
        Preconditions.checkState(StringUtils.isNotBlank(businessKey), "businessKey不能为空");
        // 特地创建一个map对象,用作启动流程实例的参数;注意要与data区分开,data是不会与流程实例打交道的,但是data的作用是用来贯穿整个方法的,专门用于数据传递;也就是说,data里面的数据不会弄脏流程实例。
        Map<String, Object> allVariables = Maps.newHashMap();
        if (!CollectionUtils.isEmpty(variables)) {
            allVariables.putAll(variables);
        }
        // processDefinitionKey是一个抽象方法,需要子类实现,不同的流程定义不一样
        String processDefinitionKey = processDefinitionKey();
        allVariables.put(Const.PROCESS_DEFINITION_KEY, processDefinitionKey);
        // 前置操作
        beforeStartProcessInstance(processDefinitionKey, businessKey, executor, allVariables, data);
        // 启动流程实例
        ProcessInstance processInstance;
        if (CollectionUtils.isEmpty(allVariables)) {
            processInstance =
                    runtimeService.startProcessInstanceByKey(processDefinitionKey, businessKey);
        } else {
            processInstance =
                    runtimeService.startProcessInstanceByKey(
                            processDefinitionKey, businessKey, allVariables);
        }
        // 后置操作
        afterStartProcessInstance(processInstance, businessKey, executor, variables, data);
        return processInstance;
    }
    // 前置操作
    protected void beforeStartProcessInstance(
            String processDefinitionKey,
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data)
            throws GlobalException {
        // 通过startEventKey获取启动节点
        UserTaskHandler userTaskHandler = userTaskHandlers.get(startEventKey());
        if (Objects.nonNull(userTaskHandler)) {
            // 设置启动节点的执行人
            identityService.setAuthenticatedUserId(executor.getId());
            // 同样的引入任务处理前置操作
            userTaskHandler.beforeTask(null, businessKey, executor, variables, data);
        }
        // 读取一些自定义配置,把配置信息存储进variables;这些配置将被用在启动流程实例,这样就可以在这个流程实例存活的任意时刻和节点获取相关配置,方便后续扩展;比如后续可以扩展动态指派任务、待办任务消息提示等业务功能
        activitiBusinessDataRepository.readProcessInstanceVariable(
                processDefinitionKey(), startEventKey(), variables);
    }
    // 后置操作
    protected void afterStartProcessInstance(
            ProcessInstance processInstance,
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data) {
        Timestamp now = new Timestamp(System.currentTimeMillis());
        String processInstanceId = processInstance.getId();
        String processDefinitionKey = processInstance.getProcessDefinitionKey();
        String processDefinitionName = processInstance.getProcessDefinitionName();
        String startUserId = processInstance.getStartUserId();
        long startTime = processInstance.getStartTime().getTime();
        TaskNodeInfo taskNodeInfo =
                TaskNodeInfo.newInstance(START_EVENT_KEY, startEventName(), startEventKey());
        // 根据任务节点taskDefinitionKey获取指定的UserTaskHandler
        UserTaskHandler userTaskHandler = userTaskHandlers.get(taskNodeInfo.getDefinitionKey());
        List<FieldInfo> processInstanceDataList = null;
        List<FieldInfo> taskNodeDataList = null;
        if (Objects.nonNull(userTaskHandler)) {
            // 获取审核结果和建议;默认用户任务都会有审核结果和建议。
            Object result = userTaskHandler.approvalResult(variables, data);
            if (Objects.nonNull(result)) {
                data.put(Approval.APPROVAL_RESULT.name(), result);
            }
            Object opinion = userTaskHandler.approvalOpinion(variables, data);
            if (Objects.nonNull(opinion)) {
                data.put(Approval.APPROVAL_OPINION.name(), opinion);
            }
            // 用户任务中,需要保存到流程实例中的数据
            processInstanceDataList =
                    userTaskHandler.processInstanceData(
                            processInstance, executor, businessKey, variables, data);
            // 用户任务中,需要保存到节点快照的数据
            taskNodeDataList =
                    userTaskHandler.taskNodeData(
                            processInstance, executor, businessKey, variables, data);
        }
        // 和上面两个对应,下面是程序自动保存的数据,不依赖于开发者
        // 自动保存的流程实例数据
        List<FieldInfo> autoSaveProcessInstanceData =
                activitiBusinessDataRepository.getAutoSaveBusinessInstanceVariables(
                        processDefinitionKey, processDefinitionName, startUserId, startTime);
        // 自动保存的流程节点数据
        List<FieldInfo> autoSaveTaskNodeVariables =
                activitiBusinessDataRepository.getAutoSaveTaskNodeVariables(
                        executor, businessKey, data);
        // 保存该流程实例数据
        activitiBusinessDataRepository.saveProcessInstanceData(
                processInstanceDataList,
                autoSaveProcessInstanceData,
                processInstanceId,
                businessKey,
                now);
        // 保存节点快照数据
        activitiBusinessDataRepository.saveTaskNodeData(
                autoSaveTaskNodeVariables, taskNodeDataList, processInstanceId, taskNodeInfo, now);
        // 保存节点历史数据;历史数据属于自动保存的,不需要开发者操心。
        activitiBusinessDataRepository.saveHistory(
                businessKey, processInstanceId, executor, taskNodeInfo, data, now);
    }
    /**
     * 所有任务办理入口
     * @description
     */
    @Transactional(rollbackFor = GlobalException.class)
    public Task completeTask(
            String taskId,
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data)
            throws GlobalException {
        // 通过taskId作为查询参数,还有一个就是当前办理人的id
        Task task = queryUnassignedTask(taskId, executor.getId());
        if (Objects.isNull(task)) {
            throw GlobalException.newInstance("TASK_NOT_EXIST", "任务不存在");
        }
        // 同上面的流程启动;作为办理任务的参数容器
        Map<String, Object> allVariables = Maps.newHashMap();
        if (!CollectionUtils.isEmpty(variables)) {
            allVariables.putAll(variables);
        }
        // 前置操作
        beforeCompleteTask(task, businessKey, executor, allVariables, data);
        // 任务办理
        if (CollectionUtils.isEmpty(allVariables)) {
            taskService.complete(task.getId());
        } else {
            taskService.complete(task.getId(), allVariables);
        }
        // 后置操作
        afterCompleteTask(task, businessKey, executor, allVariables, data);
        return task;
    }
    // 前置操作
    protected void beforeCompleteTask(
            Task task,
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data)
            throws GlobalException {
        // 获取当前任务的taskDefinitionKey
        String taskDefinitionKey = task.getTaskDefinitionKey();
        // 查询目标UserTaskHandler
        UserTaskHandler userTaskHandler = userTaskHandlers.get(taskDefinitionKey);
        if (Objects.nonNull(userTaskHandler)) {
            // 执行任务处理的前置操作
            userTaskHandler.beforeTask(task, businessKey, executor, variables, data);
        }
        // 读取该节点相关配置;只有执行到该节点,才会读取该节点的配置,而不是一次性读取流程的所有配置;注意按需加载
        activitiBusinessDataRepository.readProcessInstanceVariable(
                processDefinitionKey(), taskDefinitionKey, variables);
    }
    // 后置操作
    protected void afterCompleteTask(
            Task task,
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data) {
        Timestamp now = new Timestamp(System.currentTimeMillis());
        String processInstanceId = task.getProcessInstanceId();
        TaskNodeInfo taskNodeInfo =
                TaskNodeInfo.newInstance(task.getId(), task.getName(), task.getTaskDefinitionKey());
        // 拿到UserTaskHandler
        UserTaskHandler userTaskHandler = userTaskHandlers.get(taskNodeInfo.getDefinitionKey());
        // 说白了,下面还是准备数据,因为后面需要把数据作持久化
        List<FieldInfo> processInstanceDataList = null;
        List<FieldInfo> taskNodeDataList = null;
        if (Objects.nonNull(userTaskHandler)) {
            // 审核意见和建议
            Object result = userTaskHandler.approvalResult(variables, data);
            if (Objects.nonNull(result)) {
                data.put(Approval.APPROVAL_RESULT.name(), result);
            }
            Object opinion = userTaskHandler.approvalOpinion(variables, data);
            if (Objects.nonNull(opinion)) {
                data.put(Approval.APPROVAL_OPINION.name(), opinion);
            }
            // 流程实例数据
            processInstanceDataList =
                    userTaskHandler.processInstanceData(
                            task, executor, businessKey, variables, data);
            // 节点数据
            taskNodeDataList =
                    userTaskHandler.taskNodeData(task, executor, businessKey, variables, data);
        }
        // 自动保存的流程节点数据
        List<FieldInfo> autoSaveTaskNodeVariables =
                activitiBusinessDataRepository.getAutoSaveTaskNodeVariables(
                        executor, businessKey, data);
        // 数据准备好后,下面开始持久化
        // 保存流程实例数据
        activitiBusinessDataRepository.saveProcessInstanceData(
                processInstanceDataList, null, processInstanceId, businessKey, now);
        // 保存节点快照数据
        activitiBusinessDataRepository.saveTaskNodeData(
                autoSaveTaskNodeVariables, taskNodeDataList, processInstanceId, taskNodeInfo, now);
        // 保存节点历史数据;其实是当前节点的时间、节点、办理人等相关信息,结合上面的快照数据,就可以完全恢复当前节点办理的任务
        activitiBusinessDataRepository.saveHistory(
                businessKey, processInstanceId, executor, taskNodeInfo, data, now);
    }
    /**
     * 指定processDefinitionKey,需要子类实现
     *
     * @return
     */
    protected abstract String processDefinitionKey();
    /**
     * 指定startEventKey
     *
     * @return
     */
    protected String startEventKey() {
        return getStartEvent().getId();
    }
    /**
     * 指定startEventName
     *
     * @return
     */
    protected String startEventName() {
        return getStartEvent().getName();
    }
    protected ProcessDefinition getProcessDefinition() {
        return this.processDefinition;
    }
    protected BpmnModel getBpmnModel() {
        return this.bpmnModel;
    }
    protected StartEvent getStartEvent() {
        return this.startEvent;
    }
    protected Collection<FlowElement> getXmlNodeElements() {
        return this.xmlNodeElements;
    }
    // 实现ExecutionListener前置操作,避免子类不需要实现的时候也要写一个空实现;子类如果真的需要实现,可以重写该方法
    @Override
    protected void beforeExecute(
            DelegateExecution execution, Map<String, Object> variables, Map<String, Object> data) {}
  // 实现ExecutionListener的notify操作,避免子类不需要实现的时候也要写一个空实现;子类如果真的需要实现,可以重写该方法
  @Override
    protected void execute(
            DelegateExecution execution, Map<String, Object> variables, Map<String, Object> data) {
    }
  // 实现ExecutionListener后置操作,避免子类不需要实现的时候也要写一个空实现;子类如果真的需要实现,可以重写该方法
    @Override
    protected void afterExecute(DelegateExecution execution, Map<String, Object> data) {
    }
    /**
     * 不能重写;在流程实例的实现中,不需要实现这个方法,只有任务处理器需要实现
     *
     * @param execution
     * @param data
     * @return
     */
    @Override
    protected final List<FieldInfo> processInstanceData(
            DelegateExecution execution, Map<String, Object> data) {
        return null;
    }
    /**
     * 不能重写;在流程实例的实现中,不需要实现这个方法,只有任务处理器需要实现
     *
     * @param execution
     * @param data
     * @return
     */
    @Override
    protected final List<FieldInfo> taskNodeData(
            DelegateExecution execution, Map<String, Object> data) {
        return null;
    }
}
复制代码
  • 大家一定要仔细看上面这段代码的注释,里面阐述了这样设计的原因和思路
  • 任务处理器
// 实现AbstractElement,开始分支出任务处理器
public abstract class AbstractTaskHandler extends AbstractElement {
    private static final long serialVersionUID = -1470919793703094859L;
    @Autowired
    private IBusinessNodeVariablesRepository businessNodeVariablesRepository;
    // 通过任务ID查询历史任务
    public Object historyTaskInfo(String taskId){
        List<TaskNodeVariables> historyDataList =
                businessNodeVariablesRepository.findByTaskId(taskId);
        if (CollectionUtils.isEmpty(historyDataList)) {
            return null;
        }
        Map<String, Object> map = Maps.newHashMap();
        for (TaskNodeVariables nodeVariables : historyDataList) {
            map.put(nodeVariables.getFieldName(), nodeVariables);
        }
        return map;
    }
}
// 注意UserTaskHandler实现了UserTaskListener,一定要回头去看ActivitiListener
public abstract class UserTaskHandler extends AbstractTaskHandler implements UserTaskListener {
    /**
     * 办理任务之前准备variables;同样的空实现,是为了避免子类一定要去写这个实现
     *
     * @param task
     * @param businessKey
     * @param executor
     * @param variables
     * @param data
     */
    public void beforeTask(
            Task task,
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data)
            throws GlobalException {
    }
    /**
     * 这个是启动流程实例和完成待办任务的实现中都会调用的函数,自定义保存节点数据;同样的空实现,是为了避免子类一定要去写这个实现
     *
     * @param obj
     * @param executor
     * @param businessKey
     * @param variables
     * @return
     */
    public List<FieldInfo> taskNodeData(
            Object obj,
            Executor executor,
            String businessKey,
            Map<String, Object> variables,
            Map<String, Object> data) {
        return null;
    }
    /**
     * 这个是启动流程实例和完成待办任务的实现中都会调用的函数,自定义保存实例数据;同样的空实现,是为了避免子类一定要去写这个实现
     *
     * @param obj
     * @param executor
     * @param businessKey
     * @param variables
     * @return
     */
    public List<FieldInfo> processInstanceData(
            Object obj,
            Executor executor,
            String businessKey,
            Map<String, Object> variables,
            Map<String, Object> data) {
        return null;
    }
    /**
     * 审批结果;这个就是要求所有的用户任务必须要有审核结果
     *
     * @return
     */
    public abstract Object approvalResult(Map<String, Object> variables, Map<String, Object> data);
    /**
     * 审批意见;要求所有的用户任务必须要有审核意见
     *
     * @return
     */
    public abstract Object approvalOpinion(Map<String, Object> variables, Map<String, Object> data);
    /**
     * Execute前置处理;同样的空实现,是为了避免子类一定要去写这个实现
     *
     * @param execution
     * @param variables
     * @param data
     */
    @Override
    protected void beforeExecute(
            DelegateExecution execution, Map<String, Object> variables, Map<String, Object> data) {
    }
    /**
     * 后置处理;同样的空实现,是为了避免子类一定要去写这个实现
     *
     * @param execution
     * @param data
     */
    @Override
    protected final void afterExecute(DelegateExecution execution, Map<String, Object> data) {
    }
    // 这个基本没用,设计上还需要调整;同样的空实现,是为了避免子类一定要去写这个实现 
    @Override
    protected final List<FieldInfo> processInstanceData(
            DelegateExecution execution, Map<String, Object> data) {
        return null;
    }
    // 这个基本没用,设计上还需要调整;同样的空实现,是为了避免子类一定要去写这个实现 
    @Override
    protected final List<FieldInfo> taskNodeData(
            DelegateExecution execution, Map<String, Object> data) {
        return null;
    }
    // 这个就是实现ExecutionListener的空实现
    @Override
    protected void execute(
            DelegateExecution execution, Map<String, Object> variables, Map<String, Object> data) {
    }
    // 如果是任务被创建的事件的话,那么先设置办理人,这就是动态实现设置办理人的思路;有需要可以扩展其他功能。
    @Override
    public void notify(DelegateTask delegateTask) {
        if (StringUtils.equalsIgnoreCase(delegateTask.getEventName(), "create")) {
            setAssignee(delegateTask);
        }
    }
    /**
     * 设置办理人
     *
     * @param delegateTask
     */
    protected void setAssignee(DelegateTask delegateTask) {
        // 获取processDefinitionKey
        String processDefinitionKey =
                CastUtil.castString(delegateTask.getVariable(Const.PROCESS_DEFINITION_KEY));
        // 获取taskDefinitionKey
        String taskDefinitionKey = delegateTask.getTaskDefinitionKey();
        Map<String, Object> variables = Maps.newHashMap();
        // 查询节点配置
        readTransactorAndSendNoticeVariable(processDefinitionKey, taskDefinitionKey, variables);
        // 根据配置查询办理人;TODO
        String assignee = "";
        delegateTask.setAssignee(assignee);
    }
  // 需要特殊处理的启动节点;我把它当用户任务处理掉
  public abstract static class StartEventTaskHandler extends UserTaskHandler {
        // 把不可能调用的直接final处理掉
        @Override
        public final void notify(DelegateTask delegateTask) {
        }
        @Override
        public List<FieldInfo> taskNodeData(
                Object obj,
                Executor executor,
                String businessKey,
                Map<String, Object> variables,
                Map<String, Object> data) {
            return null;
        }
        @Override
        public List<FieldInfo> processInstanceData(
                Object obj,
                Executor executor,
                String businessKey,
                Map<String, Object> variables,
                Map<String, Object> data) {
            return null;
        }
        // 把不可能调用的直接final处理掉
        @Override
        public final Object approvalResult(Map<String, Object> variables, Map<String, Object> data) {
            return null;
        }
        // 把不可能调用的直接final处理掉
        @Override
        public final Object approvalOpinion(Map<String, Object> variables, Map<String, Object> data) {
            return null;
        }
    }
  // 需要特殊处理的结束节点;我把它当用户任务处理掉
  public abstract static class EndEventTaskHandler extends UserTaskHandler {
        // 把不可能调用的直接final处理掉
        @Override
        public final List<FieldInfo> taskNodeData(
                Object obj,
                Executor executor,
                String businessKey,
                Map<String, Object> variables,
                Map<String, Object> data) {
            return null;
        }
        // 把不可能调用的直接final处理掉
        @Override
        public final List<FieldInfo> processInstanceData(
                Object obj,
                Executor executor,
                String businessKey,
                Map<String, Object> variables,
                Map<String, Object> data) {
            return null;
        }
        // 把不可能调用的直接final处理掉
        @Override
        public final Object approvalResult(Map<String, Object> variables, Map<String, Object> data) {
            return null;
        }
        // 把不可能调用的直接final处理掉
        @Override
        public final Object approvalOpinion(Map<String, Object> variables, Map<String, Object> data) {
            return null;
        }
        // 把不可能调用的直接final处理掉
        @Override
        public final void notify(DelegateTask delegateTask) {
        }
    }
}
// 这个类和ActivitiListener类似的原理,但是是针对ServiceTask的,因为ServiceTask组件需要指定一个JavaDelegate的实现
public class ServiceTaskDelegate implements JavaDelegate {
    @Override
    public void execute(DelegateExecution execution) {
        // 获取组件id
        String taskDefinitionKey = execution.getCurrentActivityId();
        // 查找ServiceTaskHandler对象
        ServiceTaskHandler serviceTaskHandler =
                ApplicationContextUtil.getBean(taskDefinitionKey, ServiceTaskHandler.class);
        if (Objects.nonNull(serviceTaskHandler)) {
            // 调用目标方法
            serviceTaskHandler.notify(execution);
        }
    }
}
// 这个是系统任务处理器,activiti里面还有一个ServiceTask的组件
public abstract class ServiceTaskHandler extends AbstractTaskHandler {
    // 重写ExecutionListener后置操作,保存数据
    @Override
    protected void afterExecute(DelegateExecution execution, Map<String, Object> data) {
        saveData(execution, data);
    }
    // 空实现
    @Override
    protected List<FieldInfo> processInstanceData(
            DelegateExecution execution, Map<String, Object> data) {
        return null;
    }
    // 空实现
    @Override
    protected List<FieldInfo> taskNodeData(DelegateExecution execution, Map<String, Object> data) {
        return null;
    }
    // 实现ExecutionListener的操作
    @Override
    protected void execute(
            DelegateExecution execution, Map<String, Object> variables, Map<String, Object> data) {
        Object result = approvalResult(execution, variables, data);
        Object opinion = approvalOpinion(execution, variables, data);
        if (Objects.nonNull(result)) {
            data.put(Approval.APPROVAL_RESULT.name(), result);
        }
        if (Objects.nonNull(opinion)) {
            data.put(Approval.APPROVAL_OPINION.name(), opinion);
        }
    }
    /**
     * 审批结果
     *
     * @return
     */
    public abstract Object approvalResult(
            DelegateExecution execution, Map<String, Object> variables, Map<String, Object> data);
    /**
     * 审批意见
     *
     * @return
     */
    public abstract Object approvalOpinion(
            DelegateExecution execution, Map<String, Object> variables, Map<String, Object> data);
}
// 网关任务处理器
public abstract class GateTaskHandler extends AbstractElement {
    private static final long serialVersionUID = -1425610003326612058L;
    // 空实现
    @Override
    protected void afterExecute(DelegateExecution execution, Map<String, Object> data) {}
    // 把不可能调用的直接final处理掉
    @Override
    protected final List<FieldInfo> processInstanceData(
            DelegateExecution execution, Map<String, Object> data) {
        return null;
    }
    // 把不可能调用的直接final处理掉
    @Override
    protected final List<FieldInfo> taskNodeData(
            DelegateExecution execution, Map<String, Object> data) {
        return null;
    }
}
复制代码
  • 按照这个,我把最主要的UserTask、ServiceTask等几个常见的任务给处理掉了。下面来看一下这些抽象类都怎么使用。

具体实现

// "leave"是流程图的id
@Service("leave")
public class LeaveServiceImpl extends BaseProcessServiceImpl
        implements ILeaveService {
  /**
     * 流程定义key
     */
    private static final String LEAVE_PROCESS_DEFINITION_KEY = "leave";
    @Override
    protected String processDefinitionKey() {
        return LEAVE_PROCESS_DEFINITION_KEY;
    }
    // 接口方法,需要实现,申请假期
    @Transactional(rollbackFor = Exception.class)
    @Override
    public String apply(Object data, String userId) throws GlobalException {
      // 准备数据TODO
      // 调用BaseProcessServiceImpl启动方法启动流程实例
      ProcessInstance processInstance =
                startProcessInstance(businessKey, Executor.defaultExecutor(userId),
                        variables, data);
        return processInstance.getId();
    }
    // 接口方法,需要实现,审批业务
    @Transactional(rollbackFor = Exception.class)
    @Override
    public String approval(Object approval, String userId)
            throws GlobalException {
        // 准备数据
        Map<String, Object> variables = handleApprovalVariables(approval, userId);
        Map<String, Object> data = handleApprovalData(approval);
        // 调用父类的办理方法
        Task task =
                completeTask(
                        approval.getTaskId(), approval.getBusinessKey(),
                        Executor.defaultExecutor(userId), variables, data);
        return task.getId();
    }
}
// 启动节点ID
@Component("leave_apply")
public class ApplyTask extends UserTaskHandler.StartEventTaskHandler {
    // 任务前置操作
    @Override
    public void beforeTask(
            Task task,
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data)
            throws GlobalException {
    }
    // 任务数据快照
    @Override
    public List<FieldInfo> taskNodeData(
            Object obj,
            Executor executor,
            String businessKey,
            Map<String, Object> variables,
            Map<String, Object> data) {
        if (CollectionUtils.isEmpty(data)) {
            return null;
        }
        FieldInfo.Builder builder = new FieldInfo.Builder();
        for (Map.Entry<String, Object> entry : data.entrySet()) {
            String value = CastUtil.castString(entry.getValue(), StringUtils.EMPTY);
            builder.normalFieldInfo(entry.getKey(), value, "");
        }
        return builder.build();
    }
    // 保存在流程实例中的数据
    @Override
    public List<FieldInfo> processInstanceData(
            Object obj,
            Executor executor,
            String businessKey,
            Map<String, Object> variables,
            Map<String, Object> data) {
        String leave_reason = CastUtil.castString(variables.get("leave_reason"), StringUtils.EMPTY);
        String leave_date = CastUtil.castString(variables.get("leave_date"), StringUtils.EMPTY);
        return new FieldInfo.Builder()
                .normalFieldInfo(
                        "leave_reason", leave_reason, "请假原因")
                .normalFieldInfo("leave_date", leave_date, "请假日期")
                .build();
    }
    @Override
    protected void execute(
            DelegateExecution execution, Map<String, Object> variables, Map<String, Object> data) {
    }
}
// 审批节点id
@Component("leave_approval")
public class UserApprovalTask extends UserTaskHandler {
    @Override
    public void beforeTask(
            Task task,
            String businessKey,
            Executor executor,
            Map<String, Object> variables,
            Map<String, Object> data)
            throws GlobalException {
    }
    // 节点快照数据;这里我存储的是审批数据
    @Override
    public List<FieldInfo> taskNodeData(
            Object obj,
            Executor executor,
            String businessKey,
            Map<String, Object> variables,
            Map<String, Object> data) {
        String approvalUserId = CastUtil.castString(variables.get("approvalUserId"), StringUtils.EMPTY);
        String approvalResult = CastUtil.castString(data.get("approvalResult"), StringUtils.EMPTY);
        String approvalOpinion = CastUtil.castString(data.get("approvalOpinion"), StringUtils.EMPTY);
        FieldInfo.Builder builder = new FieldInfo.Builder();
        builder.normalFieldInfo("approvalUserId", approvalUserId, "审批人ID")
                .normalFieldInfo("approvalResult", approvalResult, "审批结果")
                .normalFieldInfo("approvalOpinion", approvalOpinion, "审批意见");
        return builder.build();
    }
    // 同样的,我把需要的数据存储进流程实例中
    @Override
    public List<FieldInfo> processInstanceData(
            Object obj,
            Executor executor,
            String businessKey,
            Map<String, Object> variables,
            Map<String, Object> data) {
        String approvalUserId = CastUtil.castString(variables.get("approvalUserId"), StringUtils.EMPTY);
        String approvalResult = CastUtil.castString(variables.get("approvalResult"), StringUtils.EMPTY);
        String approvalOpinion = CastUtil.castString(variables.get("approvalOpinion"), StringUtils.EMPTY);
        return new FieldInfo.Builder()
                .normalFieldInfo(
                        "approvalUserId",
                        approvalUserId,
                        "审批人ID")
                .normalFieldInfo("approvalResult",
                        approvalResult,
                        "审批结果")
                .normalFieldInfo(
                        "approvalOpinion",
                        approvalOpinion,
                        "审批意见")
                .build();
    }
    // 审批结果
    @Override
    public Object approvalResult(Map<String, Object> variables, Map<String, Object> data) {
        return data.getOrDefault("approvalResult", StrUtil.EMPTY);
    }
    // 审批意见
    @Override
    public Object approvalOpinion(Map<String, Object> variables, Map<String, Object> data) {
        return data.getOrDefault("approvalOpinion", StrUtil.EMPTY);
    }
    // 自定义设置办理人
    @Override
    protected void setAssignee(DelegateTask delegateTask) {
        Object approvalUserId = delegateTask.getVariable("approvalUserId");
        if (Objects.isNull(approvalUserId)) {
            super.setAssignee(delegateTask);
        } else {
            String userId = CastUtil.castString(approvalUserId);
            delegateTask.setAssignee(userId);
        }
    }
}
复制代码

通过以上代码及相关注释,我已经把我对于activiti工作量引擎的使用与思考总结起来了,感兴趣的小伙伴可以参与一起讨论。


相关文章
|
存储 SQL 消息中间件
大数据生态圈常用组件(一):数据库、查询引擎、ETL工具、调度工具等
大数据生态圈常用组件(一):数据库、查询引擎、ETL工具、调度工具等
|
7月前
|
运维 Serverless 文件存储
Serverless 应用引擎产品使用合集之是基于什么逻辑运行的
阿里云Serverless 应用引擎(SAE)提供了完整的微服务应用生命周期管理能力,包括应用部署、服务治理、开发运维、资源管理等功能,并通过扩展功能支持多环境管理、API Gateway、事件驱动等高级应用场景,帮助企业快速构建、部署、运维和扩展微服务架构,实现Serverless化的应用部署与运维模式。以下是对SAE产品使用合集的概述,包括应用管理、服务治理、开发运维、资源管理等方面。
|
8月前
|
人工智能 供应链 监控
推荐一款TinyEngine低代码引擎!支持自定义DSL 生成定制的源码、跨技术栈!
推荐一款TinyEngine低代码引擎!支持自定义DSL 生成定制的源码、跨技术栈!
150 0
|
XML Java 关系型数据库
Activiti工作流引擎介绍
Activiti工作流引擎介绍
|
8月前
|
边缘计算 安全 算法
阿里云丁玉杰:构建全场景服务引擎
2023全球边缘计算大会·上海站,阿里云边缘云演讲分享
194 0
|
数据库
易搭工作流引擎用是什么开源 还是阿里自研产品,零代码平台场景页面映射数据库表是动态创建,采用什么框架处理,怎么让系统产生高并发能力。易搭权限有没有了解,求解。
易搭工作流引擎用是什么开源 还是阿里自研产品,零代码平台场景页面映射数据库表是动态创建,采用什么框架处理,怎么让系统产生高并发能力。易搭权限有没有了解,求解。
|
存储 XML SQL
Activiti工作流框架学习笔记(一)之通用数据表详细介绍
Activiti工作流框架学习笔记(一)之通用数据表详细介绍
539 1
|
存储 XML Java
Activiti工作流
Activiti工作流
159 0
|
存储 XML Java
Activiti工作流与业务整合实战
Activiti工作流与业务整合实战
558 0
Activiti工作流与业务整合实战
|
XML 存储 物联网
OneCode低代码引擎-流程引擎白皮书
在低代码应用中,应用比例非常高的一种应用便是以流程+表单驱动为模型的各种审批类引用。但流程在低代码平台中的应用绝不是简简单单的流程+表单的模型。而是站在更高的层次上在自然时间轴为基础的维度上,将事件、数据、响应、人工交互等因素进行特定场景下的编排逻辑处理。
OneCode低代码引擎-流程引擎白皮书