spring boot flowable多人前加签

博客围绕Spring Boot后端开发展开,主要介绍了前加签插件以及Controller处理相关内容,为后端开发提供了一定的技术参考。

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

1、前加签插件

package com.xxx.flowable.cmd;

import com.xxx.auth.security.user.SecurityUser;
import com.xxx.commons.ApplicationContextHolder;
import com.google.common.collect.Lists;
import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import org.flowable.common.engine.impl.interceptor.CommandContext;
import org.flowable.engine.TaskService;
import org.flowable.engine.impl.cmd.NeedsActiveTaskCmd;
import org.flowable.task.service.impl.persistence.entity.TaskEntity;
import org.flowable.task.service.impl.persistence.entity.TaskEntityImpl;

import java.util.*;

/**
 *
 * @author HanKeQi
 * @date 2023/9/25
 */
public class BeforeDelegateTaskCmd extends NeedsActiveTaskCmd<Void> {


    private List<String> assignees;


    public BeforeDelegateTaskCmd(String taskId, Set<String> assignees) {
        super(taskId);
        this.assignees = Lists.newArrayList(assignees);
    }


    @Override
    protected Void execute(CommandContext commandContext, TaskEntity taskEntity) {
        if (CollectionUtils.isNotEmpty(this.assignees)){
        	// 根据自己方法回去bean
            TaskService taskService = ApplicationContextHolder.getBean(TaskService.class);
            assert taskService != null;
            TaskEntityImpl currentTask = (TaskEntityImpl)taskEntity;
            String parentTaskId = currentTask.getParentTaskId();
            //获取当前操作人,也可以传入
            String userId = SecurityUser.getUserId();
            if (StringUtils.isEmpty(parentTaskId)){
                currentTask.setOwner(userId);
                currentTask.setAssignee(null);
                currentTask.setCountEnabled(true);
                currentTask.setScopeType("before");
                taskService.saveTask(taskEntity);
                parentTaskId = taskEntity.getId();
            }

            for (String assignee: this.assignees) {
                String uuid = UUID.randomUUID().toString();
                TaskEntity task = (TaskEntity)taskService.newTask(uuid);
                task.setCategory(currentTask.getCategory());
                task.setDescription(currentTask.getDescription());
                task.setTenantId(currentTask.getTenantId());
                task.setOwner(userId);
                task.setAssignee(assignee);
//                task.setExecutionId(currentTask.getExecutionId());
                task.setName(String.format("加签%s", currentTask.getName()));
                task.setParentTaskId(currentTask.getId());
                task.setProcessDefinitionId(currentTask.getProcessDefinitionId());
                task.setProcessInstanceId(currentTask.getProcessInstanceId());
                task.setTaskDefinitionKey(String.format("dy_%s", currentTask.getTaskDefinitionKey()));
//                task.setPriority(currentTask.getPriority());
                task.setCreateTime(new Date());

                task.setStatus(currentTask.getStatus());
                task.setItemCode(currentTask.getItemCode());
                task.setApproveType(currentTask.getApproveType());
                task.setBussId(currentTask.getBussId());
                task.setBussName(currentTask.getBussName());


                taskService.saveTask(task);
            }

            //如果是候选人,需要删除运行时候选表种的数据。
            long candidateCount = taskService.createTaskQuery().taskId(parentTaskId).taskCandidateUser(userId).count();
            if (candidateCount > 0) {
                taskService.deleteCandidateUser(parentTaskId, userId);
            }

        }

        return null;
    }

}

2、Controller 处理

package com.xxx.flowable.controller;

import lombok.AllArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.flowable.engine.TaskService;
import org.flowable.task.api.Task;
import org.springframework.util.CollectionUtils;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;
import java.util.Set;

/**
 * @author HanKeQi
 * @date 2023/9/28
 */
@Slf4j
@RestController
@AllArgsConstructor
@RequestMapping("flowable/test")
public class Test001Controller {

    private final TaskService taskService;

    @PostMapping
    public String submit(String taskId, Set<String> assignees){
        Task task = taskService.createTaskQuery().taskId(taskId).singleResult();
        Task parentTask = null;
        String parentTaskId = task.getParentTaskId();
        if (StringUtils.isNotEmpty(parentTaskId)){
            parentTask = taskService.createTaskQuery().taskId(parentTaskId).singleResult();
            List<Task> subTasks = taskService.getSubTasks(parentTaskId);
            //处理最后一个的时候,一定要把任务返还给加签人
            if (!CollectionUtils.isEmpty(subTasks) && subTasks.size() == 1) {
                taskService.resolveTask(parentTaskId);
            }
        }
        //如果要获取变量请使用parentTask 获取传输变量
        return "ok";
    }
}

### Flowable 工作流引擎实现批功能 Flowable 是一种轻量级的工作流和业务流程管理 (BPM) 平台,支持 BPMN 2.0 标准以及 CMMN 和 DMN 的建模。它能够帮助开发者快速构建复杂的应用程序逻辑并集成到现有系统中[^1]。 #### 批功能概述 批功能通常用于审批流程,在企业应用中非常常见。通过 Flowable 可以定义一个步骤的审批过程,其中每个步骤可以由不同的角色完成特定的任务。以下是关于如何利用 Flowable 来实现批功能的一些核心概念和技术要点: --- #### 定义流程模型 在 Flowable 中,可以通过 BPMN 图形化工具来设计流程图。对于批场景来说,常见的需求可能包括个节点之间的流转、条件判断(如是否同意)、会模式等。这些都可以通过配置 `User Task` 节点及其属性来满足。 ```xml <process id="approvalProcess" name="Approval Process"> <!-- 开始事件 --> <startEvent id="startEvent"/> <!-- 用户任务:提交申请 --> <userTask id="submitApplication" name="Submit Application" assignee="${initiator}"/> <!-- 用户任务:领导审核 --> <userTask id="managerReview" name="Manager Review" candidateGroups="managers"/> <!-- 排他网关:决定下一步走向 --> <exclusiveGateway id="decisionPoint"/> <!-- 结束事件 --> <endEvent id="approvedEnd" name="Approved"/> <endEvent id="rejectedEnd" name="Rejected"/> <!-- 流程连线 --> <sequenceFlow sourceRef="startEvent" targetRef="submitApplication"/> <sequenceFlow sourceRef="submitApplication" targetRef="managerReview"/> <sequenceFlow sourceRef="managerReview" targetRef="decisionPoint"/> <sequenceFlow sourceRef="decisionPoint" targetRef="approvedEnd" conditionExpression="${approved == true}"/> <sequenceFlow sourceRef="decisionPoint" targetRef="rejectedEnd" conditionExpression="${approved != true}"/> </process> ``` 上述 XML 片段展示了基本的批流程结构,其中包括了启动事件、用户任务、排他网关以及结束事件等个组件[^2]。 --- #### 配置服务端环境 为了运行基于 Flowable 的应用程序,需要设置好 Java 运行时环境,并引入必要的依赖库。Maven 构建文件中的部分片段如下所示: ```xml <!-- Maven POM 文件中的 Flowable 依赖项 --> <dependencies> <dependency> <groupId>org.flowable</groupId> <artifactId>flowable-spring-boot-starter-process</artifactId> <version>6.7.2</version> </dependency> <dependency> <groupId>com.h2database</groupId> <artifactId>h2</artifactId> <scope>runtime</scope> </dependency> </dependencies> ``` 此代码片段说明了项目所需的核心依赖关系,确保 Spring Boot 应用能顺利Flowable 功能模块[^3]。 --- #### 编写自定义逻辑 除了标准的流程定义外,还可以扩展额外的功能以适应更复杂的业务需求。例如,当某个环节涉及共同评审时,则需启用 **实例子任务** 或者采用 **Ad-Hoc Sub-Processes** 技术方案。 下面是一个简单的例子展示如何动态调整参与者列表: ```java // 设置候选集合 List<String> reviewers = Arrays.asList("john", "jane"); taskService.setAssignee(task.getId(), reviewers.get(0)); // 分配给第一个成员处理 if (!reviewers.isEmpty()) { taskService.addCandidateUsers(task.getId(), new HashSet<>(reviewers.subList(1, reviewers.size()))); } ``` 这里演示的是将初始任务指派给指定员之后再将其余潜在执行者入候选池的操作方法[^4]。 --- #### 数据持久化与查询优化 考虑到实际生产环境中可能会存在大量历史记录数据积累的情况,因此建议合理规划数据库索引策略以便提高检索效率。另外也可以考虑定期清理已完成的老版本实例从而减少存储压力。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值