权限管理系统-0.6.0

七、员工端审批

员工端审批的大致流程如下图:
在这里插入图片描述
这个模块目的是实现员工在微信端的审批提交和处理功能,为了与之前的管理系统区分开,新建一个controller完成这些功能。

7.1 查询审批分类和审批模板

7.1.1 后端接口

//controller
@Api(tags = "员工端审批")
@RestController
@RequestMapping("/admin/process")
//@CrossOrigin注解用于实现跨域,之前管理页面在前端配置了跨域,微信端在控制器上实现跨域
@CrossOrigin
public class WChatProcessController {

    @Autowired
    private OaProcessTypeService processTypeService;

    /**
     * 查询审批类型和对应模板
     */
    @ApiOperation("查询审批类型和对应模板")
    @GetMapping("/getAllProcessTypeAndTemplate")
    public Result getAllProcessTypeAndTemplate(){
        return Result.ok(processTypeService.getAllProcessTypeAndTemplate());
    }

}
//ProcessTypeService
    /**
     * 查询所有审批类型和对应模板
     * @return
     */
    @Override
    public List<ProcessType> getAllProcessTypeAndTemplate() {
        //查询所有审批类型
        List<ProcessType> processTypes = baseMapper.selectList(null);
        //遍历审批类型
        for(ProcessType processType : processTypes){
            //根据审批类型的id查询审批模板
            LambdaQueryWrapper<ProcessTemplate> processTemplateLambdaQueryWrapper = new LambdaQueryWrapper<>();
            processTemplateLambdaQueryWrapper.eq(ProcessTemplate::getProcessTypeId,processType.getId());
            List<ProcessTemplate> processTemplates = processTemplateMapper.selectList(processTemplateLambdaQueryWrapper);
            //将结果保存在processType中
            processType.setProcessTemplateList(processTemplates);
        }
        return processTypes;
    }

7.1.2 前端

  1. 首先将本文所带资源下载后和管理端前端放在一个目录下。该文件包含了静态资源页面,直接使用。导入了这个文件后,只需要做一点小小修改就能使用了,大部分代码已经写好了。
    在这里插入图片描述
  2. 下载依赖:npm install
  3. 更改配置:src/utils/request.js文件
    在这里插入图片描述
  4. 将App.vue文件中如图内容注释:
    在这里插入图片描述
  5. 使用命令npm run serve运行项目进行测试:
    在这里插入图片描述

7.2 审批申请

当用户选择了模板要提交审批信息时,需要从后端查询模板相应信息并返回给前端。

    /**
     * 根据id查询模板信息
     */
    @ApiOperation("查询模板信息")
    @GetMapping("/getProcessTemplate/{id}")
    public Result getProcessTemplate(@PathVariable Long id){
        return Result.ok(processTemplateService.getById(id));
    }

在这里插入图片描述
在这里插入图片描述
可以看到表单信息是以JSON形式发送给后端的。

7.3 启动流程实例

  1. 先创建一个工具类,将当前登录人的id和name通过ThreadLocal与当前线程绑定起来,方便在其他地方使用登陆人信息。
/**
 * 获取当前用户信息帮助类
 */
public class UserInfoHelper {

    private static ThreadLocal<Long> userId = new ThreadLocal<Long>();
    private static ThreadLocal<String> username = new ThreadLocal<String>();

    public static void setUserId(Long _userId) {
        userId.set(_userId);
    }
    public static Long getUserId() {
        return userId.get();
    }
    public static void removeUserId() {
        userId.remove();
    }
    public static void setUsername(String _username) {
        username.set(_username);
    }
    public static String getUsername() {
        return username.get();
    }
    public static void removeUsername() {
        username.remove();
    }
}
//接着修改TokenAuthenticationFilter中的getAuthentication方法,获取用户名和id后,将他们存入线程变量
    private UsernamePasswordAuthenticationToken getAuthentication(HttpServletRequest request){
        String token = request.getHeader("token");
        if(!StringUtils.isEmpty(token)){
            //获取用户名
            String username = JWTHelper.getUsername(token);
            //从redis中查询用户权限
            String s = (String) redisTemplate.opsForValue().get(username);
            //将用户名和id存为线程变量
            UserInfoHelper.setUserId(JWTHelper.getUserId(token));
            UserInfoHelper.setUsername(username);
            //将字符串转换为对象
            List<SimpleGrantedAuthority> simpleGrantedAuthorities = JSON.parseArray(s, SimpleGrantedAuthority.class);
            return new UsernamePasswordAuthenticationToken(username,null, simpleGrantedAuthorities);
        }
        return null;
    }
  1. 启动审批流程controller接口:
//ProcessFormVo:用于提交表单信息
@Data
@ApiModel(description = "流程表单")
public class ProcessFormVo {

    @ApiModelProperty(value = "审批模板id")
    private Long processTemplateId;

    @ApiModelProperty(value = "审批类型id")
    private Long processTypeId;

    @ApiModelProperty(value = "表单值")
    private String formValues;

}
//controller
    /**
     * 启动审批流程
     */
    @ApiOperation("启动审批流程")
    @PostMapping("/startUp")
    public Result startUp(@RequestBody ProcessFormVo processFormVo){
        processService.startUp(processFormVo);
        return Result.ok();
    }
  1. 启动审批流程的service方法:
    /**
     * 启动流程实例
     * @param processFormVo
     */
    @Override
    public void startUp(ProcessFormVo processFormVo) {
        //1.查询用户信息
        SysUser sysUser = sysUserMapper.selectById(UserInfoHelper.getUserId());
        //2.查询模板信息
        ProcessTemplate processTemplate = processTemplateMapper.selectById(processFormVo.getProcessTemplateId());
        //3.封装要向oa_process添加的数据
        Process process = new Process();
        BeanUtils.copyProperties(processFormVo,process);
        process.setProcessCode(System.currentTimeMillis()+"");
        process.setUserId(UserInfoHelper.getUserId());
        process.setFormValues(processFormVo.getFormValues());
        process.setTitle(sysUser.getName() + "发起" + processTemplate.getName() + "申请");
        //状态:0:默认  1:审批中  2:审批完成  -1:驳回
        process.setStatus(1);
        processMapper.insert(process);
        //4.启动流程实例
        //得到业务id和表单数据
        String businessKey = process.getId().toString();
        String formValues = processFormVo.getFormValues();
        //将formValues转换成JSONObject
        JSONObject jsonObject = JSON.parseObject(formValues);
        JSONObject formData = jsonObject.getJSONObject("formData");
        //JSONObject类似MAp,所以可以使用遍历map的方式遍历,再将得到的数据存在map中
        HashMap<String, Object> stringObjectHashMap = new HashMap<>();
        for(Map.Entry<String,Object> entry : formData.entrySet()){
            stringObjectHashMap.put(entry.getKey(),entry.getValue());
        }
        //启动流程实例
        ProcessInstance processInstance = runtimeService.startProcessInstanceByKey(processTemplate.getProcessDefinitionKey(),
                businessKey, stringObjectHashMap);
        //业务表关联流程实例id
        process.setProcessInstanceId(processInstance.getProcessInstanceId());
        //5.查询下一个审批人,有可能是多个
        //根据流程实例id获取当前任务,有可能返回多条任务,因为流程可能有分支
        List<Task> list = taskService.createTaskQuery().processInstanceId(processInstance.getProcessInstanceId()).list();
        //遍历任务列表,得到审批人名单
        ArrayList<String> strings = new ArrayList<>();
        for(Task task : list){
            LambdaQueryWrapper<SysUser> sysUserLambdaQueryWrapper = new LambdaQueryWrapper<>();
            sysUserLambdaQueryWrapper.eq(SysUser::getUsername,task.getAssignee());
            SysUser sysUser1 = sysUserMapper.selectOne(sysUserLambdaQueryWrapper);
            strings.add(task.getAssignee());
            //6.推送消息给审批人:后续实现
        }
        //设置流程描述信息
        process.setDescription("等待"+ StringUtils.join(strings.toArray(),',')+"审批");
        //向oa_process添加数据
        processMapper.insert(process);
    }

7.4 保存审批记录

  1. 建表,用于记录每个审批流程的每一步操作:
CREATE TABLE `oa_process_record` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'id',
  `process_id` bigint(20) NOT NULL DEFAULT '0' COMMENT '审批流程id',
  `description` varchar(255) DEFAULT NULL COMMENT '审批描述',
  `status` tinyint(3) DEFAULT '0' COMMENT '状态',
  `operate_user_id` bigint(20) NOT NULL DEFAULT '0' COMMENT '操作用户id',
  `operate_user` varchar(20) DEFAULT NULL COMMENT '操作用户',
  `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
  `update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
  `is_deleted` tinyint(3) NOT NULL DEFAULT '0' COMMENT '删除标记(0:不可用 1:可用)',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8 COMMENT='审批记录';
  1. 新建一个实体类,用于记录每个审批流程的每一步操作:
@Data
@ApiModel(description = "ProcessRecord")
@TableName("oa_process_record")
public class ProcessRecord extends BaseEntity {

	private static final long serialVersionUID = 1L;

	@ApiModelProperty(value = "审批流程id")
	@TableField("process_id")
	private Long processId;

	@ApiModelProperty(value = "审批描述")
	@TableField("description")
	private String description;

	@ApiModelProperty(value = "状态")
	@TableField("status")
	private Integer status;

	@ApiModelProperty(value = "操作用户id")
	@TableField("operate_user_id")
	private Long operateUserId;

	@ApiModelProperty(value = "操作用户")
	@TableField("operate_user")
	private String operateUser;

}
  1. 使用代码生成器,生成相应代码,删掉生成的控制器,因为对操作的记录由后端来完成,不需要和前端交互;
  2. 定义service方法:
/**
 * <p>
 * 审批记录 服务实现类
 * </p>
 *
 * @author beiluo
 * @since 2024-03-21
 */
@Service
public class OaProcessRecordServiceImpl extends ServiceImpl<OaProcessRecordMapper, ProcessRecord> implements OaProcessRecordService {

    @Autowired
    private SysUserMapper sysUserMapper;

    @Autowired
    private OaProcessRecordMapper processRecordMapper;

    @Override
    public void processOperationRecord(Long processId, Integer status, String description) {
        //首先创建一个ProcessRecord实例
        ProcessRecord processRecord = new ProcessRecord();
        //获取操作用户
        SysUser sysUser = sysUserMapper.selectById(UserInfoHelper.getUserId());
        //向对象中添加数据
        processRecord.setProcessId(processId);
        processRecord.setStatus(status);
        processRecord.setDescription(description);
        processRecord.setOperateUserId(sysUser.getId());
        //下面添加用户的真实姓名而不是用户名
        processRecord.setOperateUser(sysUser.getName());
        //保存数据到数据库
        processRecordMapper.insert(processRecord);
    }
}
  1. 修改启动流程实例方法:
//加入下面代码
        //向oa_process_record添加数据
        processRecordService.processOperationRecord(process.getId(),process.getStatus(),process.getDescription());

7.5 审批人查询待办任务

  1. 控制器方法:
    /**
     * 审批人查询待办任务
     */
    @ApiOperation("审批人查询待办任务")
    @GetMapping("/getToDoList/{page}/{limit}")
    public Result getToDoList(@PathVariable Long page, @PathVariable Long limit){
        Page<ProcessVo> processVoPage = new Page<>(page, limit);
        return Result.ok(processService.getToDoList(processVoPage));
    }
  1. service方法
    /**
     * 审批人查询待办任务
     * @param processVoPage
     * @return
     */
    @Override
    public IPage<ProcessVo> getToDoList(Page<ProcessVo> processVoPage) {
        //获取审人待办任务
        TaskQuery asc = taskService.createTaskQuery().taskAssignee(UserInfoHelper.getUsername())
                .orderByTaskCreateTime()
                .asc();
        //第一个参数为记录起始位置,第二个参数为每页记录数
        //getCurrent表示页码,getSize表示每页记录数
        List<Task> tasks = asc.listPage((int) ((processVoPage.getCurrent() - 1) * processVoPage.getSize()), (int) processVoPage.getSize());
        //遍历任务列表,将其转换成processVo的列表
        ArrayList<ProcessVo> processVos = new ArrayList<>();
        for(Task task : tasks){
            //首先获得流程id
            String processInstanceId = task.getProcessInstanceId();
            //通过流程实例id查询相应流程
            ProcessInstance processInstance = runtimeService.createProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();
            //通过流程实例获取业务id,也就是oa_process表的主键
            String businessKey = processInstance.getBusinessKey();
            //通过主键查询process表
            Process process = processMapper.selectById(businessKey);
            //向processVo中添加数据
            ProcessVo processVo = new ProcessVo();
            BeanUtils.copyProperties(process,processVo);
            processVo.setTaskId(task.getId());
            processVos.add(processVo);
        }
        IPage<ProcessVo> processVoPage1 = new Page<>(processVoPage.getCurrent(), processVoPage.getSize(), tasks.size());
        return processVoPage1.setRecords(processVos);
    }

7.6 测试

  1. 首先启动项目;
  2. 接着进入到审批模板添加页面,添加一个新的模板:
    在这里插入图片描述
    在这里插入图片描述
    注意在上传文件这里,对文件的命名有要求:压缩包的名称必须为流程的DefinitionKey,以便后端获取,xml文件的名称中间必须加上.bpmn20,命名可以是压缩包名.bpmn20.xml。流程图的绘制在Activiti7入门这篇文章中有。
    这里上传文件的时候有个bug,注意controller接口中的上传方法的参数名一定要与前端发送的数据名一致,也就是形参MultipartFile的变量名要为file,否则会接收不到上传文件。
    在这里插入图片描述
    并且之前这个流程定义上传接口没写完全,还需要在方法结尾加上如下代码:
        Map<String, Object> map = new HashMap<>();
        //根据上传地址后续部署流程定义,文件名称为流程定义的默认key
        map.put("processDefinitionPath", "processes/" + originalFilename);
        map.put("processDefinitionKey", originalFilename.substring(0, originalFilename.lastIndexOf(".")));
        return Result.ok(map);

上传流程是:前端填写好信息并上传文件后,后端接口将文件放在指定位置后,返回流程定义键和流程定义文件存放路径给前端,之后前端将本次模板的相关信息封装发送给后端,后端调用接口将模板信息存储在数据库中。
3. 然后点击发布:
在这里插入图片描述
4. 接着在oa-web/views/test.vue页面进行修改,将如下图中的token改成绘制的流程图中指定负责人的token,这个负责人必须存在与数据库中。再将之前用于测试在request.js中添加的token删掉。
在这里插入图片描述
5. 修改完成后,访问test页面,进行测试:
在这里插入图片描述
6. 先选择admin提交请假申请,在测试页点击admin之后,返回http://localhost:9090/#/页面,刷新后提交请假申请:
在这里插入图片描述

  1. 接着在测试页面点击张三,然后返回http://localhost:9090/#/页面,刷新后就能看到需要审批的任务:
    在这里插入图片描述
    这里因为我把获取审批任务的接口写在了OaProcessController中,所以在请求时出现了跨域问题,只需要在类上添加@CrossOrigin注解就可以解决这个问题了。
    这里需要提醒一下注意检查前端api的请求路径中的名称与自己的controller上的请求路径是否一致。

7.7 显示审批详情

创建一个接口用于返回审批的详细信息:

    /**
     *
     * @param id 流程实例id
     * @return
     */
    @ApiOperation("返回审批详细信息")
    @GetMapping("showProcessDetails/{id}")
    public Result showProcessDetails(@PathVariable Long id){
        return Result.ok(processService.getProcessDetails(id));
    }
//OaProcessServiceImpl方法
    @Override
    public Map<String, Object> getProcessDetails(Long id) {
        //根据流程实例id获取实例信息
        Process process = processMapper.selectById(id);
        //有流程实例id获取当前流程操作记录
        LambdaQueryWrapper<ProcessRecord> processRecordLambdaQueryWrapper = new LambdaQueryWrapper<>();
        processRecordLambdaQueryWrapper.eq(ProcessRecord::getProcessId,id);
        List<ProcessRecord> list = processRecordService.list(processRecordLambdaQueryWrapper);
        //获取模板信息
        Long processTemplateId = process.getProcessTemplateId();
        ProcessTemplate processTemplate = processTemplateMapper.selectById(processTemplateId);
        //判断当前用户是否可以审批
            //先获取当前流程实例下的所有任务
        List<Task> list1 = taskService.createTaskQuery().processInstanceId(process.getProcessInstanceId()).list();
        //如果有任务的负责人是当前用户,那么就设置标记为true
        boolean isApprove = false;
        for (Task task : list1) {
            if(task.getAssignee().equals(UserInfoHelper.getUsername())){
                isApprove = true;
                break;
            }
        }
        //将数据封装在map中返回
        HashMap<String, Object> stringObjectHashMap = new HashMap<>();
        stringObjectHashMap.put("process",process);
        stringObjectHashMap.put("processRecordList",list);
        stringObjectHashMap.put("processTemplate",processTemplate);
        stringObjectHashMap.put("isApprove",isApprove);
        return stringObjectHashMap;
    }

在这里插入图片描述

7.8 审批任务

//先创建一个审批条件类,用于接收前端的数据
@Data
public class ApprovalVo {

    private Long processId;

    private String taskId;

    @ApiModelProperty(value = "状态")
    private Integer status;

    @ApiModelProperty(value = "审批描述")
    private String description;
}
//控制器方法
    /**
     * 审批任务
     */
    @ApiOperation("审批任务")
    @PostMapping("approve")
    public Result approveTask(ApprovalVo approvalVo){
        processService.approveTask(approvalVo);
        return Result.ok();
    }
    /**
     * 审批任务
     * @param approvalVo
     */
    @Override
    public void approveTask(ApprovalVo approvalVo) {
        //获取任务id
        String taskId = approvalVo.getTaskId();
        String assignee = taskService.createTaskQuery().taskId(taskId).singleResult().getAssignee();
        if(approvalVo.getStatus() == 1){
            //如果状态值为1,说明审批通过
            taskService.complete(taskId);
        }else{
            this.endTask(taskId);
        }
        //记录操作
        processRecordService.processOperationRecord(approvalVo.getProcessId(),
                approvalVo.getStatus(),
                assignee+(approvalVo.getStatus()==1?"通过了":"驳回了")+"申请");

        //查询下一个审批人
        Process process = processMapper.selectById(approvalVo.getProcessId());
        List<Task> list = taskService.createTaskQuery().processInstanceId(process.getProcessInstanceId()).list();
        if(CollectionUtils.isEmpty(list)){
            //如果集合为空,则说明流程已经审批结束
            if(approvalVo.getStatus()==1){
                process.setDescription("审评通过");
                process.setStatus(2);
            }else{
                process.setDescription("审批驳回");
                process.setStatus(-1);
            }
        }else{
            //如果任务列表不为空,则查询下一个负责人
            ArrayList<String> strings = new ArrayList<>();
            for (Task task : list) {
                LambdaQueryWrapper<SysUser> sysUserLambdaQueryWrapper = new LambdaQueryWrapper<>();
                sysUserLambdaQueryWrapper.eq(SysUser::getUsername,task.getAssignee());
                SysUser sysUser = sysUserMapper.selectOne(sysUserLambdaQueryWrapper);
                //添加用户真实姓名
                strings.add(sysUser.getName());

                //推送消息
            }
            process.setStatus(1);
            process.setDescription("等待" + StringUtils.join(strings.toArray(), ",") + "审批");
        }
        processMapper.updateById(process);
    }

    private void endTask(String taskId) {
        //查询当前任务
        Task task = taskService.createTaskQuery().taskId(taskId).singleResult();
        //查询bpmn模型,参数为流程定义id
        BpmnModel bpmnModel = repositoryService.getBpmnModel(task.getProcessDefinitionId());
        //获取结束事件列表
        List<EndEvent> endEventList = bpmnModel.getMainProcess().findFlowElementsOfType(EndEvent.class);
        //假设只有一个结束事件,下面获得结束节点
        FlowNode endEvent = endEventList.get(0);
        //获取当前节点
        FlowNode currentFlow = (FlowNode) bpmnModel.getMainProcess().getFlowElement(task.getTaskDefinitionKey());
        //清理当前节点的流向
        currentFlow.getOutgoingFlows().clear();
        //创建新的流向
        SequenceFlow sequenceFlow = new SequenceFlow();
        sequenceFlow.setId("newSequenceFlow");
        sequenceFlow.setSourceFlowElement(currentFlow);
        sequenceFlow.setTargetFlowElement(endEvent);
        ArrayList<SequenceFlow> sequenceFlows = new ArrayList<>();
        sequenceFlows.add(sequenceFlow);
        //向当前节点加入下面流程
        currentFlow.setOutgoingFlows(sequenceFlows);

        //完成当前任务
        taskService.complete(taskId);
    }
//再show.vue的approve方法中加入如下代码
  let approvalVo = {
    processId: this.process.id,
    taskId: this.taskId,
    status: status
  }
  api.approve(approvalVo).then(response => {
    this.$router.push({ path: '/list/1' })
  })

这里解决一个bug,审批任务的controller接口方法的参数要加上@RequestBody注解,因为数据是以JSON形式发送的。

7.9 已处理任务

//控制器方法
    /**
     * 查询已处理任务
     */
    @ApiOperation("查询已处理任务")
    @GetMapping("/getApprovedTasks/{page}/{limit}")
    public Result getApprovedTasks(@PathVariable Long page,@PathVariable Long limit){
        Page<ProcessVo> processVoPage = new Page<>(page, limit);
        return Result.ok(processService.getApprovedTasks(processVoPage));
    }
//service方法
    /**
     * 查询已处理任务
     * @param processVoPage
     * @return
     */
    @Override
    public IPage<ProcessVo> getApprovedTasks(Page<ProcessVo> processVoPage) {
        //首先跟据用户名获取已处理任务列表
        List<HistoricTaskInstance> historicTaskInstances = historyService.createHistoricTaskInstanceQuery()
                .taskAssignee(UserInfoHelper.getUsername())
                .finished()
                .orderByTaskCreateTime()
                .desc()
                .listPage((int) ((processVoPage.getCurrent() - 1) * processVoPage.getSize()), (int) processVoPage.getSize());

        int count = historicTaskInstances.size();
        //得到历史任务后,遍历任务列表,将其转换为返回信息
        ArrayList<ProcessVo> processVos = new ArrayList<>();
        for (HistoricTaskInstance historicTaskInstance : historicTaskInstances) {
            //根据任务id得到process
            LambdaQueryWrapper<Process> processLambdaQueryWrapper = new LambdaQueryWrapper<>();
            processLambdaQueryWrapper.eq(Process::getProcessInstanceId,historicTaskInstance.getProcessInstanceId());
            Process process = processMapper.selectOne(processLambdaQueryWrapper);
            ProcessVo processVo = new ProcessVo();
            BeanUtils.copyProperties(process,processVo);
            processVos.add(processVo);
        }
        //封装返回对象
        IPage<ProcessVo> processVoIPage = new Page<ProcessVo>(processVoPage.getCurrent(), processVoPage.getSize(), count);
        processVoIPage.setRecords(processVos);
        return processVoIPage;
    }

在这里插入图片描述

7.10 已提交任务

//控制器方法
    /**
     * 查询已提交任务,已提交任务包括未完成的和已完成的
     */
    @ApiOperation("查询已提交审批")
    @GetMapping("/getSubmittedTasks/{page}/{limit}")
    public Result getSubmittedTasks(@PathVariable Long page,@PathVariable Long limit){
        Page<ProcessVo> processVoPage = new Page<>(page, limit);
        return Result.ok(processService.getSubmittedTasks(processVoPage));
    }
//service方法
    @Override
    public IPage<ProcessVo> getSubmittedTasks(Page<ProcessVo> processVoPage) {
        //调用ProcessMapper的selectPageList方法实现
        //首先设置查询条件
        ProcessQueryVo processQueryVo = new ProcessQueryVo();
        processQueryVo.setUserId(UserInfoHelper.getUserId());
        //根据查询条件调用mapper方法
        IPage<ProcessVo> processVoIPage = processMapper.selectPageList(processVoPage,processQueryVo);
        return processVoIPage;
    }

在这里插入图片描述

7.11 获取当前登录用户信息

    /**
     * 获取当前登录用户信息
     */
    @ApiOperation("获取当前登录用户信息")
    @GetMapping("/getCurrentUser")
    public Result getCurrentUser(){
        SysUser byId = sysUserService.getById(UserInfoHelper.getUserId());
        //封装结果并返回
        HashMap<String, Object> stringObjectHashMap = new HashMap<>();
        stringObjectHashMap.put("name",byId.getName());
        stringObjectHashMap.put("phone",byId.getPhone());
        return Result.ok(stringObjectHashMap);
    }

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【论文速读】| 视觉对抗样本:突破对齐的大语言模型

本次分享论文为&#xff1a;Visual Adversarial Examples: Jailbreak Aligned Large Language Models 基本信息 原文作者&#xff1a;Xiangyu Qi, Peter Henderson, Kaixuan Huang, Ashwinee Panda, Mengdi Wang, Prateek Mittal 作者单位&#xff1a;普林斯顿大学、斯坦福大…

使用npm创建一个全局的cli命令,就像vue-cli一样

我们用过vue-cli等工具包&#xff0c;全局安装之后&#xff0c;我们可以直接使用vue create等命令&#xff0c;实际上能够这样使用的原因&#xff0c;就是使用了package.json里面的bin字段注册命令。接下来就以一个脚本文件为例子为大家演示一下bin是如何发挥作用的。 创建项目…

数据库引论:2.SQL简介

SQL(Structured Query Language,结构化查询语言) 2.1 SQL查询语言概览 SQL语言包含 数据定义语言(Data-Definition Language,DDL)。SQL DDL提供定义关系模式、删除关系以及修改关系模式的命令。数据操纵语言(Data-Manipulation Language,DML)。SQL DML提供从数据库中查询信息…

PLC通过智能网关采用HTTP协议JSON文件对接MES等服务系统平台

智能网关IGT-DSER集成了多种PLC的原厂协议&#xff0c;方便实现各种PLC、智能仪表通过HTTP协议与MES等各种系统平台通讯对接。PLC内不用编写程序&#xff0c;通过网关的参数配置软件(下载地址)绑定JSON文件的字段与PLC寄存器地址&#xff0c;配置URL即可。支持POST/GET/PUT等多…

基于YOLOv5s的电动车入梯识别系统(数据集+权重+登录界面+GUI界面+mysql)

本人训练的yolov5s模型&#xff0c;准确率在98.6%左右&#xff0c;可准确完成电梯内检测电动车任务&#xff0c;并搭配了GUI检测界面&#xff0c;支持权重选择、图片检测、视频检测、摄像头检测、识别结果拍照和在线标注数据集等功能。 并且为用户提供了登录注册功能&#xff0…

python版:使用TotalSegmentator工具可在1分钟内自动分割全身117个器官,附批量技巧

TotalSegmentator用于对 CT 图像中超过 117 个类别进行分割的工具。它接受了各种不同 CT 图像&#xff08;不同扫描仪、机构、协议等&#xff09;的训练&#xff0c;因此应该适用于大多数图像。大部分训练数据集可以从Zenodo下载&#xff08;1228 个主题&#xff09;。您还可以…

由浅到深认识Java语言(7):方法(函数)

该文章Github地址&#xff1a;https://github.com/AntonyCheng/java-notes 在此介绍一下作者开源的SpringBoot项目初始化模板&#xff08;Github仓库地址&#xff1a;https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址&#xff1a;https://blog.c…

ZYNQ EMIO MIO

1 概述 先来了解GPIO的BANK分布&#xff0c;在UG585文档GPIO一章中可以看到GPIO是有4个BANK&#xff0c; 注意与MIO的BANK区分。 BANK0 控制32个信号&#xff0c;BANK1控制22个信号&#xff0c;总共是MIO的54个引脚&#xff0c;也就是诸如 SPI,I2C,USB,SD 等 PS 端外设接口&am…

【二进制求公约数】【数学】【数论】2543. 判断一个点是否可以到达

本文涉及知识点 二进制求公约数 LeetCode2543. 判断一个点是否可以到达 给你一个无穷大的网格图。一开始你在 (1, 1) &#xff0c;你需要通过有限步移动到达点 (targetX, targetY) 。 每一步 &#xff0c;你可以从点 (x, y) 移动到以下点之一&#xff1a; (x, y - x) (x - y…

基于python+vue灾害应急救援平台flask-django-php-nodejs

灾害应急救援平台的目的是让使用者可以更方便的将人、设备和场景更立体的连接在一起。能让用户以更科幻的方式使用产品&#xff0c;体验高科技时代带给人们的方便&#xff0c;同时也能让用户体会到与以往常规产品不同的体验风格。 与安卓&#xff0c;iOS相比较起来&#xff0c;…

(二)RabbitMQ实战——rabbitmq高可用集群搭建

前言 本节内容是关于rabbitmq高可用集群的部署搭建&#xff0c;使用的是centos7系统&#xff0c;我们准备三台服务器作为rabbitmq的高可用服务器&#xff0c;rabbitmq集群本身不是天然支持高可用的&#xff0c;我们通过配置rabbitmq服务器的镜像队列&#xff0c;以确保消息可以…

突然发现!原来微信批量自动加好友这么简单!

你知道如何更好地管理和利用微信资源&#xff0c;实现客户拓展和沟通吗&#xff1f;下面就教大家一招&#xff0c;帮助大家实现统一管理多个微信号以及批量自动加好友。 想要统一管理多个微信号&#xff0c;不妨试试微信管理系统&#xff0c;不仅可以多个微信号同时登录&#…

无插件网页视频播放器,支持图像绘制(包含方格子、方框等),支持音视频播放、支持录像截图,提供源码下载

前言 本播放器内部采用jessibuca插件接口&#xff0c;支持录像、截图、音视频播放等功能。播放器播放基于ws流&#xff0c;图像绘制操作&#xff1a;1&#xff09;支持绘制方格子&#xff0c;用于监控移动检测画框&#xff1b;2&#xff09;支持绘制不透明方框&#xff0c;用于…

如何进行设备的非对称性能测试

非对称性能测试介绍 RFC2544是RFC组织提出的用于评测网络互联设备&#xff08;防火墙、IDS、Switch等&#xff09;的国际标准。主要是对RFC1242中定义的性能评测参数的具体测试方法、结果的提交形式作了较详细的规定。标准中定义了4个重要的参数&#xff1a;吞吐量&#xff08…

【No.12】蓝桥杯可撤销并查集|查找|合并|撤销(C++)

前置知识 蓝桥杯并查集|路径压缩|合并优化|按秩合并|合根植物(C)-CSDN博客 可撤销并查集 关键注意 可撤销并查集的撤销功能如何实现可撤销并查集能不能用路径压缩 可撤销并查集(Reversible Union-Find)是一种扩展了标准并查集(Union-Find)数据结构的数据结构&#xff0c;它允…

Python螺旋折线蓝桥杯(来源lanqiao.cn 题目176) 时间超限

题目描述 如图所示的螺旋折线经过平面上所有整点恰好一次。 对于整点(X, Y)&#xff0c;我们定义它到原点的距离dis(X, Y)是从原点到(X, Y)的螺旋折线段的长度。 例如dis(0, 1)3, dis(-2, -1)9 给出整点坐标(X, Y)&#xff0c;你能计算出dis(X, Y)吗&#xff1f; 输入格式 …

【Unity】层(Layer)详解

1.什么是Layer? 我们在做游戏开发的时候&#xff0c;尤其是场景比较复杂的时候&#xff0c;我们就需要使用Layer来分类。 比如&#xff1a; 排除不被灯光照亮的Layer 射线检测特定的 Layer 摄像机只能看到某些 Layer 对象之间的碰撞检测 Layer … 2.添加Layer ①在Inspecto…

GZ083 产品艺术设计赛题第十

全国职业院校技能大赛 产品艺术设计赛项赛题十 赛项名称 产品艺术设计 英语名称 Product Art Design 赛项编号 GZ083 归属产业 数字产业 任务名称 “绣羽鸣春”鸟形象主题文具收纳袋设计 赛项组别 中职组 高职组 □学生组 □教师组 □师生联队试点赛项 R学生组 …

Echarts地图之——如何给地图添加背景图片

上期我们已经给地图添加了一个阴影3d的效果&#xff0c;但是背景纯色的感觉还是不怎么好看&#xff0c;希望能给地图加个背景图。 一般来说给地图加背景图的情况较少&#xff0c;加个渐变色或者根据数据的情况给某些省份设置不一样的背景色&#xff0c;这样的做法是比较多的。…

C++关键字:const

文章目录 一、const的四大作用1.修饰 变量、数组2.修饰 函数的形参、修饰 引用 (最常用&#xff09;3.修饰 指针&#xff1a;常量指针、指针常量 、只读指针4.修饰 类的成员函数、修饰 类的对象 一、const的四大作用 1.修饰 变量、数组 1.const修饰变量&#xff1a; 被const修…