Java与MySQL的精准结合:打造高效审批流程

1流程思路分析

审批流程🐱‍💻

1.串行流程

当前节点审批完成后,下一次节点才能进行操作,例如经理通过之后,总监才能审批;

图片

2.并行流程

一个审批节点需要多人联审。一般有两种方式:会签、或签;

图片

会签:

注意:别跟串行审批混淆。会签的多个角色是平行的,没有上下级关系、前后顺序。

例如:在某项文件、合同、决策或其他重要事项上,需要多个相关人员进行确认、批准或签署的情况下,参与该过程的人员同时批准,这时会签就能很大程度上提高效率,顺便也能解决人情世故的问题~~。

会签支持并签:

业务模式支持:"全部投票"、"按投票比例"、"按投票通过比例"、支持设置 "百分比" 和 "分数" 两种方式。

或签(也有叫"竞签"、"串签"):指同一个审批节点设置多个人,如ABC三人,三人会同时收到审批,只要其中任意一人审批即可到下一审批节点;

3.条件流程

根据不同规则,走不同流程

图片

例如报销5000以下直属领导审批就行,5000~10000需要总监审批。

4.混合流程

就是上述的流程混合运用;

图片

例如请假的正常流程是到直属领导审批,3天以上需要总监审批。5天以上需要总裁审批。

审批动作🙋‍♀️

  1. 通过: 通过申请,流转到下一人。原因选填。

  2. 驳回: 驳回到发起人、驳回到上一环节、驳回到之前的某一环节

  3. 保存: 保存当前填写的信息。

  4. 转交: 转交给某人审批

  5. 加签: 处理者可以在审批时临时增加审核人员

  6. 抄送: 抄送人会知晓审核过程,但不进行操作

执行动作👩‍🎓

申请: 针对当前业务表单信息发起申请,开始审批流程的传递。分为2部分:由申请提交人手动发起申请、由程序自动判断满足触发规则的数据自动发起申请;另外还要注意的2点:是否允许提交人撤销(是、否)、记录编辑(不可编辑、管理员可编辑、管理员和审批人都可编辑 );

通过: 当前步骤处理通过,进入下一步骤,若为末步骤,则流程处理完成;

退回: 将步骤退回至上一步骤,即返回至上一处理人处,若为首步骤,则不进行退回;

否决: 将步骤直接结束,执行结束动作拒绝活动,不再进行操作,或者回退至第一步骤;

撤回: 若当前步骤已处理,下一处理人未处理的情况下可进行撤回操作。

图片

节点状态🕵️‍♀️

提交人: 未提交、已提交、处理中、已结束。

处理人: 待处理、已处理。

创建业务工单: 提交人登录系统之后,选择想要发起的业务工单,填写业务工单。填写完成可以选择提交或者保存;

未提交: 列表中显示所有保存但未提交的业务工单,未提交列表中的业务工单都没有业务工单编号(唯一),状态为“未提交”。提交人可以选择某个业务工单后能方便对其进行修改、删除和提交;

已提交: 列表显示所有已经提交的业务工单,有业务工单编号,并显示(未处理)。提交后的业务工单进入处理流程,提交人无法进行修改和删除;

处理中: 列表显示所有提交并已经有节点处理的业务工单,提交人可以查询某个业务工单的处理进度;

已结束: 列表显示所有已经处理完成的业务工单;

待处理: 列表显示所有待处理业务工单;

已处理: 列表显示所有当前处理人已处理的业务工单,既被当前处理人流转到下一个流程节点的业务工单;

2UI预览🛹

图片

图片

图片

图片

图片

3表设计🚧

流程配置表🚗

CREATE TABLE `approval_config` (`approval_config_id` varchar(255) COLLATE utf8mb4_general_ci NOT NULL COMMENT '雪花算法',`corp_id` varchar(100) COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '企业ID',`node_name` varchar(100) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '节点名称',`form_id` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '表单ID 属于那个表单的流程',`approval_auto_pass` tinyint(1) DEFAULT '0' COMMENT '审批节点人员相同时自动通过[1:true;0:false]',`current_node_approver_type` tinyint DEFAULT NULL COMMENT '审批人类型[1:发起人自己;2:指定角色;3:指定人员;4:表单内人员];注:1/3/4存userid、2存roleId',`current_node_approver` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '待审批人如果是角色就是角色ID,如果是其他则是userId格式:[xxx,xxx]',`care_of` tinyint(1) DEFAULT NULL COMMENT '允许审批人转交[1:true;2:false]',`approval_type` tinyint DEFAULT NULL COMMENT '审批类型[1:会签:2:或签]',`approval_node` int DEFAULT NULL COMMENT '审批节点[取值:1~N 从小到大;0:发起节点 、1:是第一个节点、 2:第二个节点 N:是第N个节点;最大的是结束节点]',`carbon_copy_recipients_type` tinyint DEFAULT '1' COMMENT '抄送人类型[1:发起人自己;2:指定角色;3:指定人员;4:表单内人员]',`carbon_copy_recipients` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '抄送人userId 数组格式',`create_time` datetime DEFAULT NULL COMMENT '规则创建时间;默认拿最新的',`group_id` varchar(255) COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '分组[标识那些节点是同一个组的]',`visible_fields` json DEFAULT NULL COMMENT '可见字段:[{"bt": false, "kj": true, "kx": true, "field": "create_time", "children": "purchase_requisition_details"}];\r\nkj:可见;kx:可写;kb:必填。如果是最后一个节点就需要单独拉一个配置',PRIMARY KEY (`approval_config_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci COMMENT='审批流程节点规则配置表';

审批流程发起表🚓

CREATE TABLE `approval_info` (`approval_info_id` varchar(255) COLLATE utf8mb4_general_ci NOT NULL COMMENT '雪花算法',`corp_id` varchar(50) COLLATE utf8mb4_general_ci NOT NULL COMMENT '企业ID',`approval_config_group_id` varchar(255) COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '规则表的groupID',`form_id` varchar(50) COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '表单ID 属于那个表单',`founder` varchar(255) COLLATE utf8mb4_general_ci NOT NULL COMMENT '发起人',`launch_time` datetime NOT NULL COMMENT '发起时间',`end_state` tinyint(1) DEFAULT '0' COMMENT '是否结束流程[1是]',`end_time` datetime DEFAULT NULL COMMENT '流程结束时间',`object_id` varchar(255) COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '关联某条数据',`process_progress` int DEFAULT '0' COMMENT '用来记录整个流程进行到那个节点',`record_node_history` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '记录历史经历节点,英文逗号分隔0,1,2,1,2,3 最后一个值和当前节点同步',PRIMARY KEY (`approval_info_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci COMMENT='审批流程发起表';

审批记录表🚕

CREATE TABLE `approval_record` (`approval_record_id` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '雪花算法',`corp_id` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '企业ID',`approval_info_id` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '审批发起表approval_info',`form_id` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '表单ID 属于那个表单的流程',`approver_user_id` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '审批人or 转交人 userId',`approver_time` datetime NOT NULL COMMENT '审批时间',`approver_opinions` varchar(255) COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '审批意见',`approver_result` tinyint NOT NULL COMMENT '审批结果[1:同意;2:拒绝:3:转交]',`transferee` varchar(50) COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '被转交人userId;approver_result=3时必填',`transferee_text` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '转交备注(此字段暂时无用)',`accessory_url` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '附件数组',`approver_node` tinyint NOT NULL COMMENT '属于个节点的审批记录',`approver_count` tinyint DEFAULT '1' COMMENT '审批次数默认1; 例:当节点B是会签,且有人同意,有人拒绝,当拒绝后会返回上一个节点A。当再次进入节点B时,用来区分,此审批记录是第一次记录还是第二次审批记录或者第N次',PRIMARY KEY (`approval_record_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci COMMENT='审批记录表';

抄送记录表🛺

CREATE TABLE `approval_carbon_copy` (`carbon_copy_id` varchar(255) COLLATE utf8mb4_general_ci NOT NULL COMMENT '雪花',`corp_id` varchar(255) COLLATE utf8mb4_general_ci NOT NULL COMMENT '企业ID',`form_id` varchar(255) COLLATE utf8mb4_general_ci NOT NULL COMMENT '属于那个表单',`approval_info_id` varchar(255) COLLATE utf8mb4_general_ci NOT NULL COMMENT '推送那条审批的ID',`user_id` varchar(255) COLLATE utf8mb4_general_ci NOT NULL COMMENT '接收抄送人',`create_time` datetime NOT NULL COMMENT '抄送时间',`read` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否已读[1:已读;0:未读]',`approval_node` tinyint NOT NULL COMMENT '那个节点抄送的',PRIMARY KEY (`carbon_copy_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci COMMENT='审批抄送记录表';

4思路👩‍🎓👩‍🎓

流程配置

1.通过流程节点规则配置表,配置每个节点的审批人、抄送人、审批规则、可见、可写、必填字段的规则等(可以给不同的表单配置不同的审批规则)。

2.配置完毕即可,每次新增数据,读取流程规则配置表,生成一条审批流程发起记录,记录发起人,发起时间,关联某条流程规则(旧数据还是使用旧流程),流程当前节点等。

3.每个节点的审批人、抄送人、可见、可写、必填字段都是根据流程规则配置表中动态配置。

5代码🥇

审批按钮

@Transactional(rollbackFor = Exception.class)
public ResultUtils approval(String token, ApprovalDTO param) {String corpId = TokenUtils.getCorpId(token);String userId = TokenUtils.getUserId(token);LocalDateTime time = LocalDateTime.now();// 找到当前数据待审批节点,是否需要自己审批,或者已经审批过ApprovalInfo info = approvalInfoMapper.selectOne(new QueryWrapper<ApprovalInfo>().eq("corp_id", corpId).eq("object_id", param.getObjectId()).eq("form_id", param.getFormId()).select("approval_info_id", "process_progress", "record_node_history", "end_state","approval_config_group_id","founder"));Assert.isFalse(info.getEndState(), ErrorMsg.msg18);Integer currentNode = info.getProcessProgress();// 当前节点// 每个节点的审批设置List<ApprovalConfig> approvalConfigs = approvalConfigMapper.selectList(new QueryWrapper<ApprovalConfig>().eq("corp_id", corpId).eq("group_id", info.getApprovalConfigGroupId()).eq("form_id", param.getFormId()).select("approval_auto_pass", "current_node_approver_type", "visible_fields","current_node_approver", "care_of", "approval_type", "approval_node", "carbon_copy_recipients_type", "carbon_copy_recipients"));Map<Integer, ApprovalConfig> configMap = approvalConfigs.stream().collect(Collectors.toMap(ApprovalConfig::getApprovalNode, ApprovalConfig -> ApprovalConfig));ApprovalConfig config = configMap.get(info.getProcessProgress());List<String> historyNode = Lists.newArrayList(info.getRecordNodeHistory().split(","));// 历史节点需要剔除最后一个historyNode.remove(historyNode.size() - 1);Bag bag = new HashBag(historyNode);int count = bag.getCount(currentNode.toString());// 这个节点历史是否出现过count = count == 0 ? 1 : (count + 1); // 用于标注这个节点是否多次审批if (param.getType() == 3) {// 转交ApprovalRecord record = new ApprovalRecord();record.setApprovalRecordId(SnowFlakeUtil.getDefaultSnowFlakeId() + "");record.setCorpId(corpId);record.setApprovalInfoId(info.getApprovalInfoId());record.setFormId(param.getFormId());record.setApproverUserId(userId);record.setApproverTime(time);record.setApproverOpinions(param.getMessage());record.setApproverResult(param.getType());record.setTransferee(param.getTransfereeUserId());if (ObjectUtil.isNotEmpty(param.getAccessoryUrl())) {record.setAccessoryUrl(JSONUtil.toJsonStr(param.getAccessoryUrl()));}record.setApproverNode(info.getProcessProgress().byteValue());record.setApproverCount(new Byte(count + ""));approvalRecordMapper.insertSelective(record);return ResultUtils.success();}ApprovalRecord records = approvalRecordMapper.selectOne(new QueryWrapper<ApprovalRecord>().eq("corp_id", corpId).eq("form_id", param.getFormId()).eq("approver_user_id", userId).eq("approval_info_id", info.getApprovalInfoId()).eq("approver_node", currentNode).eq("approver_count", count));Assert.isFalse(ObjectUtil.isNotEmpty(records) && StrUtil.isNotBlank(records.getTransferee()), ErrorMsg.msg29);Assert.isFalse(ObjectUtil.isNotEmpty(records), ErrorMsg.msg16);// 查询是否是被转交人List<String> transferees = approvalRecordMapper.findTransferee(corpId, param.getFormId(), info.getApprovalInfoId(), currentNode, count);if (CollUtil.isEmpty(transferees)) {// 1.是否需要自己审批if (config.getCurrentNodeApproverType() == 2) {// 指定角色// 查询自己拥有那些角色List<UserRole> userRoles = userRoleMapper.selectList(new QueryWrapper<UserRole>().eq("user_id", userId).eq("corp_id", corpId).select("role_id"));List<String> roleIds = userRoles.stream().map(key -> key.getRoleId()).collect(Collectors.toList());List<Integer> currentNodeApproverRoleId = JSONUtil.toList(config.getCurrentNodeApprover().toString(), Integer.class);ArrayList<String> integers = new ArrayList<>(roleIds);boolean b = integers.retainAll(currentNodeApproverRoleId);// 存在交集则返回trueAssert.isFalse(!b, ErrorMsg.msg17);} else if (config.getCurrentNodeApproverType()==3){// 指定人员 用户IDList<String> currentNodeApproverUserId = JSONUtil.toList(config.getCurrentNodeApprover().toString(), String.class);boolean contains = currentNodeApproverUserId.contains(userId);Assert.isFalse(!contains, ErrorMsg.msg17);} else if (config.getCurrentNodeApproverType() == 1) {// 发起人自己Assert.isFalse(!info.getFounder().equals(userId), ErrorMsg.msg17);}}if (param.getType() == 2) {// 拒绝approvalRefuse(info, approvalConfigs, param, userId, corpId, count, time);return ResultUtils.success();}// 判断了需要自己审批之后,先编辑数据,然后在进行审批操作JSONObject paramData = param.getMainData();approvalEditData(corpId, paramData, time, userId, param.getFormId());// 进行审批操作if (param.getType() == 1) {// 同意approvalAgreeing(currentNode, info, configMap, param, userId, corpId, time, param.getObjectId());}return ResultUtils.success();
}

审批按钮-拒绝

/*** 审批按钮-拒绝** @param info            当前数据的审批详情* @param approvalConfigs 审批规则* @param param           参数* @param userId          当前审批人id* @param corpId          企业id* @param count           审批次数*/
private void approvalRefuse(ApprovalInfo info, List<ApprovalConfig> approvalConfigs, ApprovalDTO param, String userId, String corpId, int count, LocalDateTime time) {// 生成审批记录ApprovalRecord record = new ApprovalRecord();record.setApprovalRecordId(SnowFlakeUtil.getDefaultSnowFlakeId() + "");record.setCorpId(corpId);record.setApprovalInfoId(info.getApprovalInfoId());record.setFormId(param.getFormId());record.setApproverUserId(userId);record.setApproverTime(time);record.setApproverOpinions(param.getMessage());record.setApproverResult(param.getType());if (ObjectUtil.isNotEmpty(param.getAccessoryUrl())) {record.setAccessoryUrl(JSONUtil.toJsonStr(param.getAccessoryUrl()));}record.setApproverNode(info.getProcessProgress().byteValue());record.setApproverCount(new Byte(count + ""));approvalRecordMapper.insertSelective(record);// 回滚上一个节点info.setProcessProgress(info.getProcessProgress() - 1);info.setRecordNodeHistory(info.getRecordNodeHistory() + "," + info.getProcessProgress());// info.setCurrentState(1);approvalInfoMapper.updateByPrimaryKeySelective(info);if (info.getProcessProgress() == 0) {// 如果回到第一个节点,数据状态则变成初始String formCode = fromConfigMapper.findFormCode(corpId, param.getFormId());commonMapper.updateDataState(formCode, 0, param.getObjectId(), corpId);}
}

审批按钮-同意

/*** 审批按钮-同意** @param currentNode 处理的当前节点* @param info        流程详情* @param configMap   配置集合 key=节点;val=配置* @param param       参数* @param userId      当前审批人userId* @param corpId      企业ID* @param time        当前时间*/
private void approvalAgreeing(Integer currentNode, ApprovalInfo info, Map<Integer, ApprovalConfig> configMap,ApprovalDTO param, String userId, String corpId, LocalDateTime time, String objectId) {// 查询这个节点以前是否审批过List<String> historyNode = Lists.newArrayList(info.getRecordNodeHistory().split(","));// 历史节点需要剔除最后一个historyNode.remove(historyNode.size() - 1);Bag bag = new HashBag(historyNode);int count = bag.getCount(currentNode.toString());// 这个节点历史是否出现过count = count == 0 ? 1 : count + 1; // 用于标注这个节点是否多次审批// 生成审批记录ApprovalRecord record2 = new ApprovalRecord();record2.setApprovalRecordId(SnowFlakeUtil.getDefaultSnowFlakeId() + "");record2.setCorpId(corpId);record2.setApprovalInfoId(info.getApprovalInfoId());record2.setFormId(param.getFormId());record2.setApproverUserId(userId);record2.setApproverTime(time);record2.setApproverOpinions(param.getMessage());record2.setApproverResult(param.getType());if (ObjectUtil.isNotEmpty(param.getAccessoryUrl())) {record2.setAccessoryUrl(JSONUtil.toJsonStr(param.getAccessoryUrl()));}record2.setApproverNode(currentNode.byteValue());record2.setApproverCount(new Byte(count + ""));approvalRecordMapper.insertSelective(record2);// 获取当前节点配置ApprovalConfig currentConfig = configMap.get(currentNode);if (currentConfig.getApprovalType() == 1) {// 1会签// 判断是否是最后一个审批的,如果不是return出去List<String> userIds = null;if (currentConfig.getCurrentNodeApproverType() == 2) {// 角色id// 查询自己拥有那些角色List<UserRole> userRoles = userRoleMapper.selectList(new QueryWrapper<UserRole>().eq("user_id", userId).eq("corp_id", corpId).select("user_id"));userIds = userRoles.stream().map(key -> key.getUserId()).collect(Collectors.toList());} else if (currentConfig.getCurrentNodeApproverType()==3){// 用户IDuserIds = JSONUtil.toList(currentConfig.getCurrentNodeApprover().toString(), String.class);} else if (currentConfig.getCurrentNodeApproverType() == 1) {// 发起人自己userIds = Arrays.asList(info.getFounder());}// 查询已经审批的人!=待审批人数则不是最后一个人// 1.查询已经审批的人数Integer count1 = approvalRecordMapper.selectCount(new QueryWrapper<ApprovalRecord>().eq("corp_id", corpId).eq("approval_info_id", info.getApprovalInfoId()).eq("form_id", param.getFormId()).eq("approver_node", currentNode).eq("approver_count", count));if (userIds.size() != count1) {return;// 不是最后一个可以退出}}Integer nextNode = currentNode + 1;// 更新流程信息info.setProcessProgress(nextNode);info.setRecordNodeHistory(info.getRecordNodeHistory() + "," + info.getProcessProgress());// info.setCurrentState(2);// 到这里如果不是或签,就是会签是最后一个人同意可以判断进入下一个节点//  判断下一个节点是不是最后一个节点,如果是可以结束节点了,不是则继续往下递归if ((configMap.keySet().size() - 1) == nextNode) {info.setEndTime(time);info.setEndState(true);//  结束流程approvalInfoMapper.updateByPrimaryKeySelective(info);// 数据状态变成生效String formCode = fromConfigMapper.findFormCode(corpId, param.getFormId());commonMapper.updateDataState(formCode, 2, objectId, corpId);return;}approvalInfoMapper.updateByPrimaryKeySelective(info);// 校验下一个配置是否需要审批ApprovalConfig nextConfig = configMap.get(nextNode);// 下一个节点的配置List<RuleVO> rule = JSONUtil.toList(JSONUtil.parseArray(nextConfig.getVisibleFields()), RuleVO.class);for (RuleVO ruleVO : rule) {if (ruleVO.getKx()) {return;// 判断是否开了可写,如果开了可写,不管有没有开相同审批人自动通过,都不能往下走}}if (nextConfig.getApprovalAutoPass()) {// 开了 审批人员相同自动通过// 1.是否需要自己审批if (nextConfig.getCurrentNodeApproverType() == 2) {// 角色id// 查询自己拥有那些角色List<UserRole> userRoles = userRoleMapper.selectList(new QueryWrapper<UserRole>().eq("user_id", userId).eq("corp_id", corpId).select("role_id"));List<String> roleIds = userRoles.stream().map(key -> key.getRoleId()).collect(Collectors.toList());List<Integer> currentNodeApproverRoleId = JSONUtil.toList(nextConfig.getCurrentNodeApprover().toString(), Integer.class);ArrayList<String> integers = new ArrayList<>(roleIds);boolean b = integers.retainAll(currentNodeApproverRoleId);// 存在交集返回trueif (!b) return;} else {// 用户IDList<String> currentNodeApproverUserId = JSONUtil.toList(nextConfig.getCurrentNodeApprover().toString(), String.class);boolean contains = currentNodeApproverUserId.contains(userId);if (!contains) return;}// 需要自己审批approvalAgreeing(nextNode, info, configMap, param, userId, corpId, time, objectId);}// approvalRecordMapper.insertSelective(record2);}

合并权限(如果最后一个节点存在多个自定义权限组,就需要合并)

/*** 合并权限组** @param roleIds      当前用户拥有的角色* @param dbPowerGroup 自定义权限组*/
private List<RuleVO> mergePowerGroup(List<String> roleIds, List<PowerGroup> dbPowerGroup) {List<RuleVO> result = new ArrayList<>();// 判断当前自定义权限组是否包含 当前用户的角色for (PowerGroup power : dbPowerGroup) {List<Integer> dbRoleIds = JSONUtil.toList(power.getRoleId().toString(), Integer.class);List<String> retainAll = new ArrayList<>(roleIds);retainAll.retainAll(dbRoleIds);if (retainAll.size() != 0) {// 存在交集,证明这个自定义权限组有当前操作用户List<RuleVO> rule = JSONUtil.toList(JSONUtil.parseArray(power.getVisibleFields()), RuleVO.class);Map<String, RuleVO> tempMapDB = rule.stream().collect(Collectors.toMap(ruleVo -> ruleVo.getChildren() + ruleVo.getField(), RuleVO -> RuleVO));if (CollUtil.isEmpty(result)) {result = rule;} else {Map<String, RuleVO> tempListMap = result.stream().collect(Collectors.toMap(ruleVo -> ruleVo.getChildren() + ruleVo.getField(), RuleVO -> RuleVO));if (tempMapDB.keySet().size() > result.size()) {// 如果B权限组字段比A权限组多,就拿多的一方进行循环for (RuleVO item : rule) {RuleVO ruleVO = tempListMap.get(item.getChildren() + item.getField());if (ruleVO.getBt()) {item.setBt(ruleVO.getBt());}if (ruleVO.getKj()) {item.setKj(ruleVO.getKj());}if (ruleVO.getKx()) {item.setKx(ruleVO.getKx());}}result = rule;} else {// 否则就拿A权限组遍历for (RuleVO item : result) {// 合并组成新的规则,如果A规则为true,B规则为false,则取true的一方RuleVO ruleDB = tempMapDB.get(item.getChildren() + item.getField());if (ruleDB.getBt()) {item.setBt(ruleDB.getBt());}if (ruleDB.getKj()) {item.setKj(ruleDB.getKj());}if (ruleDB.getKx()) {item.setKx(ruleDB.getKx());}}}}}}return result;
}

最后说一句(求关注!别白嫖!)

如果这篇文章对您有所帮助,或者有所启发的话,求一键三连:点赞、转发、在看。

关注公众号:woniuxgg,在公众号中回复:笔记  就可以获得蜗牛为你精心准备的java实战语雀笔记,回复面试、开发手册、有超赞的粉丝福利!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/2778073.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

尚硅谷 Vue3+TypeScript 学习笔记(中)

目录 三、路由 3.1. 【对路由的理解】 3.2. 【基本切换效果】 3.3. 【两个注意点】 3.4.【路由器工作模式】 3.5. 【to的两种写法】 3.6. 【命名路由】 3.7. 【嵌套路由】 3.8. 【路由传参】 query参数 params参数 3.9. 【路由的props配置】 3.10. 【 replace属性…

C++,stl,list容器详解

目录 1.list基本概念 2.list构造函数 3.list的赋值和交换 4.list大小操作 5.list的插入的删除 6.list数据存取 7.list反转和排序 排序案例 1.list基本概念 2.list构造函数 #include<bits/stdc.h> using namespace std;void print(const list<int> &lk) …

.NET命令行(CLI)常用命令

本文用于记录了.NET软件开发全生命周期各阶段常用的一些CLI命令&#xff0c;用于开发速查。 .NET命令行&#xff08;CLI&#xff09;常用命令 项目创建&#xff08;1&#xff09;查看本机SDK&#xff08;2&#xff09;查看本机可以使用的.NET版本&#xff08;3&#xff09;生成…

【开源】SpringBoot框架开发桃花峪滑雪场租赁系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 游客服务2.2 雪场管理 三、数据库设计3.1 教练表3.2 教练聘请表3.3 押金规则表3.4 器材表3.5 滑雪场表3.7 售票表3.8 器材损坏表 四、系统展示五、核心代码5.1 查询教练5.2 教练聘请5.3 查询滑雪场5.4 滑雪场预定5.5 新…

基于完全二叉树实现线段树-- [爆竹声中一岁除,线段树下苦踌躇]

文章目录 一.完全二叉树完全二叉树的父子结点引索关系 二.线段树三.基于完全二叉树实现线段树关于线段树的结点数量问题的证明递归建树递归查询区间和递归单点修改线段树模板题 一.完全二叉树 完全二叉树的物理结构是线性表,逻辑结构是二叉树 完全二叉树的父子结点引索关系 …

【数据结构】双向链表(链表实现+测试+原码)

前言 在双向链表之前&#xff0c;如果需要查看单链表来复习一下&#xff0c;链接在这里&#xff1a; http://t.csdnimg.cn/Ib5qS 1.双向链表 1.1 链表的分类 实际中链表的结构非常多样&#xff0c;以下情况组合起来就有8种链表结构&#xff1a; 1.1.1 单向或者双向 1.1.2 …

专业135+总400+中国科学院大学859国科大信号与系统考研经验电子信息与通信,真题,大纲,参考书

今年考研专业课859信号与系统135&#xff0c;总分400上岸国科大&#xff0c;总结一下自己这一年的复习经验&#xff0c;希望对后面报考中科院大学的同学有所帮助。 专业课&#xff1a; 国科大不同研究所都是统一命题&#xff0c;859信号与系统的参考书目是郑君里的《信号与系…

C语言——oj刷题——调整数组使奇数全部都位于偶数前面

题目&#xff1a; 输入一个整数数组&#xff0c;实现一个函数&#xff0c;来调整该数组中数字的顺序使得数组中所有的奇数位于数组的前半部分&#xff0c;所有偶数位于数组的后半部分。 一、实现方法&#xff1a; 当我们需要对一个整数数组进行调整&#xff0c;使得奇数位于数…

[职场] 职场上该如何和同事相处呢?七种方法教你和同事友好相处 #其他#媒体

职场上该如何和同事相处呢&#xff1f;七种方法教你和同事友好相处 在职场上&#xff0c;如何和同事相处是一堂必修课。同事&#xff0c;是我们天天必须看到的人&#xff0c;只有和同事友好相处&#xff0c;我们才能生活得更好&#xff0c;工作得更好。那么&#xff0c;我们在…

Days 27 ElfBoard 板 AltiumDesigner 相同电路快速布局布线

在进行设计开发的时候&#xff0c;总会遇到相同的电路&#xff0c;或者模块&#xff0c;这些电路可以使用相同的布局和走线&#xff0c;例如 DC-DC 电源、网口 PHY 电路部分。这类型的电路&#xff0c;我们可以采用AltiumDesigner 中的 Room 进行布局和布线的快速复制&#xff…

备战蓝桥杯---动态规划之背包问题引入

先看一个背包问题的简单版&#xff1a; 如果我们暴力枚举可能会超时。 但我们想一想&#xff0c;我们其实不关心怎么放&#xff0c;我们关心的是放后剩下的体积。 用可行性描述即可。 于是我们令f[i][j]表示前i个物品能否放满体积为j的背包。 f[i][j]f[i-1][j]||f[i-1][j-v…

【十四】【C++】list 的常见用法

list 的初始化和遍历 /*list的初始化和遍历*/ #if 1 #include <list> #include <vector> #include <iostream> #include<algorithm> using namespace std;void TestList1(){list<int> L1;list<int> L2(10, 5);vector<int> v{1,2,3,4…

使用Arduino UNO和蓝牙模块制作智能小车

目录 概述 1 硬件结构 1.1 硬件组成 1.2 蓝牙模块介绍 1.3 控制板IO引脚定义 2 机械结构 3 固件设计 4 App设计 5 参考文献 概述 本文主要介绍使用Arduino UNO作为主板&#xff0c;用于控制电机和接收蓝牙模块数据。蓝牙模块用于从手机App上接收控制信号&#xff0c;使…

(每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理第10章 项目进度管理(四)

博主2023年11月通过了信息系统项目管理的考试&#xff0c;考试过程中发现考试的内容全部是教材中的内容&#xff0c;非常符合我学习的思路&#xff0c;因此博主想通过该平台把自己学习过程中的经验和教材博主认为重要的知识点分享给大家&#xff0c;希望更多的人能够通过考试&a…

巴尔加瓦算法图解:算法运用(上)

目录 树反向索引傅立叶变换 并行算法MapReduce函数 树 如果能将用户名插入到数组的正确位置就好了&#xff0c;这样就无需在插入后再排序。为此&#xff0c;有人设计了一种名为二叉查找树(binary search tree)的数据结构。 每个node的children 都不大于两个。对于其中的每个…

python_蓝桥杯刷题记录_笔记_全AC代码_入门4

题单目录 1.P1914 小书童——凯撒密码 2.P1028 [NOIP2001 普及组] 数的计算 3.P1036 [NOIP2002 普及组] 选数 4.P1149 [NOIP2008 提高组] 火柴棒等式 5.P1217 [USACO1.5] 回文质数 Prime Palindromes 6.P1478 陶陶摘苹果&#xff08;升级版&#xff09; 7.P1618 三连击&…

软件价值11-简单计算器

用python的tkinter做的简单计算器 代码&#xff1a; import tkinter as tkdef button_click(item):global expressionexpression expression str(item)input_text.set(expression)def button_clear():global expressionexpression ""input_text.set(""…

51单片机编程应用(C语言):串口通信

目录 通信的基本概念和种类 1.1串行通信与并行通信 ​编辑 1.2同步通信与异步通信 1.3单工&#xff0c;半双工&#xff0c;全双工 1.4通信速率 二、波特率和比特率的关系 串口通信简介&#xff1a; 1.接口标准 RS-232 2、D型9针接口定义 3.通信协议&#xff1a; …

嵌入式系统的前景:未来智能汽车

&#xff08;本文为简单介绍&#xff0c;个人的观点仅供参考&#xff09; 智能汽车时代已经来临!未来十年,我们的汽车将变得越来越智能化。各大汽车公司在研发自动驾驶技术,目标是实现真正的无人驾驶。要实现这一目标,嵌入式系统将发挥关键作用。 简单来说,嵌入式系统就是在汽…

【Make编译控制 01】程序编译与执行

目录 一、编译原理概述 二、编译过程分析 三、编译动静态库 四、执行过程分析 一、编译原理概述 make&#xff1a; 一个GCC工具程序&#xff0c;它会读 makefile 脚本来确定程序中的哪个部分需要编译和连接&#xff0c;然后发布必要的命令。它读出的脚本&#xff08;叫做 …