权限管理系统-0.6.0

七、员工端审批

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

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

7.1.1 后端接口

//controller
@Api(tags = "员工端审批")
@RestController
@RequestMapping("/admin/process")
//@CrossOrigin注解用于实现跨域,之前管理页面在前端配置了跨域,微信端在控制器上实现跨域
@CrossOrigin
public class WChatProcessController {@Autowiredprivate OaProcessTypeService processTypeService;/*** 查询审批类型和对应模板*/@ApiOperation("查询审批类型和对应模板")@GetMapping("/getAllProcessTypeAndTemplate")public Result getAllProcessTypeAndTemplate(){return Result.ok(processTypeService.getAllProcessTypeAndTemplate());}}
//ProcessTypeService/*** 查询所有审批类型和对应模板* @return*/@Overridepublic 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*/@Overridepublic 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转换成JSONObjectJSONObject 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);//业务表关联流程实例idprocess.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 {@Autowiredprivate SysUserMapper sysUserMapper;@Autowiredprivate OaProcessRecordMapper processRecordMapper;@Overridepublic 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*/@Overridepublic 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){//首先获得流程idString 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<>();//根据上传地址后续部署流程定义,文件名称为流程定义的默认keymap.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方法@Overridepublic 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();//如果有任务的负责人是当前用户,那么就设置标记为trueboolean 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*/@Overridepublic void approveTask(ApprovalVo approvalVo) {//获取任务idString 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模型,参数为流程定义idBpmnModel 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*/@Overridepublic 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得到processLambdaQueryWrapper<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方法@Overridepublic 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);}

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

elementUI Tree 树形控件单选实现

文章目录 展示效果代码实现elementui Tree树形控件其他详细数据 在Element UI中&#xff0c;树形控件&#xff08;el-tree&#xff09;本身不支持单选功能。但是&#xff0c;你可以通过监听节点点击事件并手动更新选中状态来实现单选树。 以下是一个简单的例子&#xff0c;展示…

【Spring 篇】SpringMVC拦截器:给你的应用增添色彩

嗨&#xff0c;亲爱的小伙伴们&#xff01;欢迎来到这段关于SpringMVC拦截器的奇妙之旅。今天我们要一探究竟&#xff0c;深入挖掘拦截器的神秘面纱&#xff0c;看看它是如何在你的应用中悄然发挥作用的。别怕&#xff0c;我会用最通俗易懂的语言&#xff0c;一步一步带你走进这…

【合合TextIn】深度解析智能文档处理技术与应用

目录 一、智能文档处理介绍 二、文档格式解析 三、图像增强技术解析 四、传统文字识别OCR技术解析 五、深度学习OCR技术解析 六、深度学习版面分析技术解析 七、文档分类 八、信息抽取 九、系统集成&#xff1a;将IDP处理后的数据集成到企业系统 结论 一、智能文档处…

下载 macOS 系统安装程序的方法

阅读信息&#xff1a; 版本&#xff1a;0.4.20231021 难度&#xff1a;1/10 到 4/10 阅读时间&#xff1a;5 分钟 适合操作系统&#xff1a;10.13, 10.14, 10.15, 11.x, 12.x&#xff0c;13.x, 14 更新2023-10-21 添加Mist的介绍支持版本的更新&#xff0c;13.x&#xff0…

Css提高——Css3的新增选择器

目录 1、Css3新增选择器列举 2、属性选择器 2.1、语法 2.2、代码&#xff1a; 2.3、效果图 3、结构伪类选择器 3.1、语法 3.2、代码 3.3、效果图 3.4、nth&#xff1a;child&#xff08;n&#xff09;的用法拓展 nth-child&#xff08;n&#xff09;与nth-of-type&#x…

MAC 帧(数据链路层)

目录 一、MAC帧的格式 二、无效的帧 三、帧间最小间隔 四、帧的发送与接收 五、小结 一、MAC帧的格式 • 常用的以太网 MAC 帧格式有两种标准 &#xff1a; DIX Ethernet V2 标准&#xff1b; IEEE 的 802.3 标准。 • 最常用的 MAC 帧是以太网 V2 的格式。 二、…

excel文件可以转成word文件吗?汇帮PDF转换器帮你实现excel转word

将Excel文件转换为Word文档是一个相对简单的任务&#xff0c;但在执行过程中需要注意一些细节&#xff0c;以确保转换后的文档格式正确、内容清晰。下面将详细介绍用汇帮PDF转换器将Excel转Word的步骤和注意事项。 一、Excel文件准备 在进行转换之前&#xff0c;首先确保Excel…

Linux部署MySQL

Linux部署MySQL5.7.17 mkdir /opt/mysql cd /opt/mysql#mysql下载官网&#xff1a; #https://downloads.mysql.com/archives/community/ #下载server、client、lib和common wget https://downloads.mysql.com/archives/get/p/23/file/mysql-community-server-5.7.17-1.el7.…

做抖店不知道怎么找达人?聊聊我是怎么找达人带货的,多看多做!

大家好&#xff0c;我是电商花花。 找不到合适的达人带货&#xff1f;不知道怎么找达人带货&#xff1f;多半都是没有用心去找达人带货&#xff0c;因为现在抖音上遍地都是达人&#xff0c;遍地都是达人在直播带货&#xff0c;在短视频带货。 而达人不是说不缺品&#xff0c;…

刚进公司第一天-电脑环境搭建

写在前面 之前在公司做过一次开发小工具的分享&#xff0c;这两天有个同事找我学习一些小工具开发的知识&#xff0c;但是我发现他的基础是真的差&#xff0c;想学开发知识却连自己本地电脑环境都没弄好&#xff0c;确实&#xff0c;有些人工作了很久&#xff0c;由于自己工作中…

了解Kafka位移自动提交的秘密:避免常见陷阱的方法

欢迎来到我的博客&#xff0c;代码的世界里&#xff0c;每一行都是一个故事 了解Kafka位移自动提交的秘密&#xff1a;避免常见陷阱的方法 前言位移自动提交简介自动提交的优缺点自动提交位移的优点&#xff1a;自动提交位移的缺点&#xff1a;自动提交与手动提交的对比分析&am…

【ScienceAI Weekly】「AI钢铁侠」黄仁勋在GTC大会聊英伟达布局科研;Overjet获牙科AI历史最大投资

AI for Science 的新成果、新动态、新视角—— 英伟达&#xff1a;有众多 AI for Scinece 领域的战略内容在布局 微软携手多家医疗机构推出 TRAIN Overjet 获牙科 AI 历史上最大的投资 联想控股与智谱 AI在生物科技等领域达成合作 Zephyr AI 获得 1.11 亿美元 A 轮融资 中…

扩容分区和文件系统(Linux)

在ECS控制台上扩容云盘容量后&#xff0c;对应分区和文件系统并未扩容&#xff0c;您还需要进入ECS实例内部继续扩容云盘的分区和文件系统&#xff0c;将扩容部分的容量划分至已有分区及文件系统内&#xff0c;使云盘扩容生效。本文为您介绍如何通过两个步骤完成Linux实例云盘的…

在线播放视频网站源码系统 带完整的安装代码包以及搭建教程

在线播放视频网站源码系统的开发&#xff0c;源于对当前视频市场的深入洞察和用户需求的精准把握。随着视频内容的爆炸式增长&#xff0c;用户对视频播放的需求也日益多样化。他们希望能够随时随地观看自己感兴趣的视频内容&#xff0c;同时还希望能够在观看过程中享受到流畅、…

【Frida】10_用鼠标自动标记棋盘上的雷区(一键过关)

&#x1f6eb; 系列文章导航 【Frida】 00_简单介绍和使用 https://blog.csdn.net/kinghzking/article/details/123225580【Frida】 01_食用指南 https://blog.csdn.net/kinghzking/article/details/126849567【Frida】02_常见API示例及功能函数封装&#xff08;snippets&#…

Go 1.22 - 更加强大的 Go 执行跟踪

原文&#xff1a;Michael Knyszek - 2024.03.14 runtime/trace 包含了一款强大的工具&#xff0c;用于理解和排查 Go 程序。这个功能可以生成一段时间内每个 goroutine 的执行追踪。然后&#xff0c;你可以使用 go tool trace 命令&#xff08;或者优秀的开源工具 gotraceui&a…

【Java - 框架 - Knife4j】(01) SpringBoot整合Knife4j - 快速上手

“SpringBoot"整合"Knife4j” - 快速上手&#xff1b; 环境 “Knife4j"版本"2.0.9”&#xff1b;“Java"版本"1.8.0_202”&#xff1b;“Spring Boot"版本"2.5.9”&#xff1b;“Windows 11 专业版_22621.2428”&#xff1b;“Intel…

GPT-5可能会在今年夏天作为对ChatGPT的“实质性改进”而到来

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

Spring Web MVC入门(6)

应用分层 在开发的过程中, 我们会发现, 程序的代码有时会很"杂乱", 如果后面的项目更大了, 那就会更加地杂乱无章(文件乱, 代码内容乱). 也基于此, 接下来让我们来学习一下应用分层. 也类似于公司的组织架构 公司初创阶段, 一个人身兼数职, 既做财务, 又做人事,还有…

Unity多人游戏基础知识总结

作者简介: 高科,先后在 IBM PlatformComputing从事网格计算,淘米网,网易从事游戏服务器开发,拥有丰富的C++,go等语言开发经验,mysql,mongo,redis等数据库,设计模式和网络库开发经验,对战棋类,回合制,moba类页游,手游有丰富的架构设计和开发经验。 (谢谢你的关注…