当前位置: 首页 > news >正文

Devops系列之对接Gerrit的设计与实现(三)-- Java编程实现

一、背景

上文讲述了如何使用shell命令实现创建gerrit项目,本文介绍如何使用java语言编程实现。

二、java语言实现

1、引入jar包

        <dependency><groupId>com.urswolfer.gerrit.client.rest</groupId><artifactId>gerrit-rest-java-client</artifactId><version>0.9.3</version></dependency>

核心类GroupApiRestClient.java

在这里插入图片描述

2、配置

project:gerrit:host: http://192.168.5.60:8080login: adminpassword: 123456# 负责人groupOwnerId: 34dd8f718972d73c9a3474b2sfad2d52x2cf596f

3、实现类

  • 配置类GerritProperties.java

import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.cloud.context.config.annotation.RefreshScope;
import org.springframework.context.annotation.Configuration;/*** gerrit配置** @author xxx*/
@RefreshScope
@Data
@Configuration
@ConfigurationProperties(prefix = "project.gerrit")
public class GerritProperties {/*** 地址*/private String host;/*** 用户名*/private String login;/*** 密码*/private String password;/*** 所属的组ID*/private String groupOwnerId;
}
  • 总体实现

import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.gerrit.extensions.api.GerritApi;
import com.google.gerrit.extensions.api.access.AccessSectionInfo;
import com.google.gerrit.extensions.api.access.PermissionInfo;
import com.google.gerrit.extensions.api.access.PermissionRuleInfo;
import com.google.gerrit.extensions.api.access.ProjectAccessInput;
import com.google.gerrit.extensions.api.groups.GroupInput;
import com.google.gerrit.extensions.api.projects.ProjectApi;
import com.google.gerrit.extensions.api.projects.ProjectInput;
import com.google.gerrit.extensions.client.SubmitType;
import com.google.gerrit.extensions.common.GroupBaseInfo;
import com.google.gerrit.extensions.common.GroupInfo;
import com.google.gerrit.extensions.common.ProjectInfo;
import com.urswolfer.gerrit.client.rest.GerritAuthData;
import com.urswolfer.gerrit.client.rest.GerritRestApiFactory;
import com.urswolfer.gerrit.client.rest.http.HttpStatusException;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;import javax.annotation.PostConstruct;
import java.util.Map;/*** gerrit api* <p>* 1.创建负责人组* 2.创建开发人组* 3.创建工程* 4.设置权限* </p>** @author xxx*/
@Slf4j
@Service
public class GerritService {private GerritRestApiFactory gerritRestApiFactory;private GerritAuthData.Basic authData;private GerritApi gerritApi;@Autowiredprivate GerritProperties gerritProperties;@PostConstructpublic void init() {gerritRestApiFactory = new GerritRestApiFactory();authData = new GerritAuthData.Basic(gerritProperties.getHost(), gerritProperties.getLogin(), gerritProperties.getPassword());gerritApi = gerritRestApiFactory.create(authData);}/*** 新建gerrit项目及组、权限** @param group       项目组* @param projectName 项目名称,比如:root/watch-answering-server*/public void create(String group, String projectName) {String ownerGroupName = projectName + "-owner";String devGroupName = projectName + "-dev";try {// 创建负责人分组GroupBaseInfo ownerGroupBaseInfo = this.createGroup(ownerGroupName, gerritProperties.getGroupOwnerId());String ownerGroupId = ownerGroupBaseInfo.id;log.info("负责人分组ID:{}", ownerGroupId);// 创建开发者分组GroupBaseInfo devGroupBaseInfo = this.createGroup(devGroupName, ownerGroupId);String devGroupId = devGroupBaseInfo.id;log.info("开发者分组ID:{}", devGroupId);// 创建工程ProjectInfo projectInfo = this.createProject(projectName, ownerGroupName);log.info("项目的详情是:{}", JsonUtils.toJsonString(projectInfo));// 赋权限项if (null != projectInfo) {this.setAccess(projectName, ownerGroupId, devGroupId);}} catch (Exception e) {log.error("创建组、工程、赋权出现了异常, projectName={}", projectName, e);}}
}
  • 创建组的实现代码
    /*** 创建组.** @param groupName 组名* @param ownerId   所属组* @return* @throws Exception*/private GroupBaseInfo createGroup(String groupName, String ownerId) throws Exception {GroupBaseInfo groupBaseInfo = new GroupBaseInfo();GroupInfo groupInfo;try {groupInfo = gerritApi.groups().id(groupName).get();log.warn("组{}已存在,请勿重复创建", groupName);} catch (HttpStatusException e) {// 不存在,则重新创建GroupInput groupInput = new GroupInput();groupInput.name = groupName;groupInput.description = groupName;groupInput.ownerId = ownerId;groupInput.visibleToAll = false;groupInfo = gerritApi.groups().create(groupInput).get();if (null == groupInfo) {throw new IllegalArgumentException("创建组出现异常");}log.info("组创建成功, groupInput={}", JsonUtils.toJsonString(groupInput));groupBaseInfo.id = groupInfo.id;groupBaseInfo.name = groupInfo.name;return groupBaseInfo;}groupBaseInfo.id = groupInfo.id;groupBaseInfo.name = groupInfo.name;return groupBaseInfo;}
  • 创建工程的实现代码
/*** 新建工程.** @param projectName    工程名* @param ownerGroupName 所属组* @return* @throws Exception*/private ProjectInfo createProject(String projectName, String ownerGroupName) throws Exception {try {ProjectApi projectApi = gerritApi.projects().name(projectName);ProjectInfo projectInfo = projectApi.get();if (null != projectInfo) {log.warn("工程{}已存在,请勿重复创建", projectName);return projectInfo;}} catch (Exception e) {ProjectInput request = new ProjectInput();request.name = projectName;request.description = projectName;request.submitType = SubmitType.INHERIT;request.owners = Lists.newArrayList(ownerGroupName);log.info("新建gerrit工程成功, request={}", JsonUtils.toJsonString(request));return gerritApi.projects().create(request).get();}return null;}
  • 赋权限的实现代码
/*** 赋权限.** @param projectName  项目名称* @param ownerGroupId 负责人组ID* @param devGroupId   开发组ID* @throws Exception*/private void setAccess(String projectName, String ownerGroupId, String devGroupId) throws Exception {try {ProjectApi projectApi = gerritApi.projects().name(projectName);ProjectAccessInput projectAccessInput = new ProjectAccessInput();Map<String, AccessSectionInfo> addMap = Maps.newHashMap();addMap.put("refs/*", buildAccessSectionInfo(ownerGroupId, devGroupId));addMap.put("refs/tags/*", buildTagsAccessSectionInfo(devGroupId));addMap.put("refs/heads/*", buildHeadsAccessSectionInfo(ownerGroupId, devGroupId));log.info("权限列表是:{}", JsonUtils.toJsonString(addMap));projectAccessInput.add = addMap;projectApi.access(projectAccessInput);} catch (HttpStatusException e) {log.error("赋权限出现异常, projectName={}, ownerGroupId={}, devGroupId={}", projectName, ownerGroupId, devGroupId, e);}}private AccessSectionInfo buildAccessSectionInfo(String ownerGroupId, String devGroupId) {AccessSectionInfo accessSectionInfo = new AccessSectionInfo();Map<String, PermissionInfo> permissions = Maps.newHashMap();permissions.put("read", buildPermissionInfo("", devGroupId));permissions.put("label-Code-Review", buildReviewPermissionInfo("Code-Review", ownerGroupId, devGroupId));permissions.put("label-Verified", buildReviewPermissionInfo("Verified", ownerGroupId, devGroupId));permissions.put("labelAs-Verified", buildReviewPermissionInfo("Verified", ownerGroupId, devGroupId));permissions.put("forgeServerAsCommitter", buildPermissionInfo("", ownerGroupId, devGroupId));permissions.put("forgeAuthor", buildPermissionInfo("", ownerGroupId, devGroupId));permissions.put("removeReviewer", buildPermissionInfo("", ownerGroupId, devGroupId));permissions.put("delete", buildPermissionInfo("", ownerGroupId));permissions.put("viewPrivateChanges", buildPermissionInfo("", ownerGroupId));permissions.put("createTag", buildPermissionInfo("", ownerGroupId));permissions.put("createSignedTag", buildPermissionInfo("", ownerGroupId));permissions.put("editHashtags", buildPermissionInfo("", ownerGroupId));permissions.put("deleteOwnChanges", buildPermissionInfo("", devGroupId));permissions.put("editTopicName", buildPermissionInfo("", devGroupId));permissions.put("abandon", buildPermissionInfo("", devGroupId));permissions.put("push", buildPermissionInfo("", devGroupId));permissions.put("addPatchSet", buildPermissionInfo("", devGroupId));permissions.put("forgeCommitter", buildPermissionInfo("", devGroupId));permissions.put("pushMerge", buildPermissionInfo("", devGroupId));permissions.put("rebase", buildPermissionInfo("", devGroupId));permissions.put("submit", buildPermissionInfo("", ownerGroupId));permissions.put("submitAs", buildPermissionInfo("", ownerGroupId));accessSectionInfo.permissions = permissions;return accessSectionInfo;}private AccessSectionInfo buildTagsAccessSectionInfo(String devGroupId) {AccessSectionInfo accessSectionInfo = new AccessSectionInfo();Map<String, PermissionInfo> permissions = Maps.newHashMap();permissions.put("createSignedTag", buildPermissionInfo("", devGroupId));permissions.put("createTag", buildPermissionInfo("", devGroupId));accessSectionInfo.permissions = permissions;return accessSectionInfo;}private AccessSectionInfo buildHeadsAccessSectionInfo(String ownerGroupId, String devGroupId) {AccessSectionInfo accessSectionInfo = new AccessSectionInfo();Map<String, PermissionInfo> permissions = Maps.newHashMap();permissions.put("forgeCommitter", buildPermissionInfo("", devGroupId));permissions.put("label-Code-Review", buildReviewPermissionInfo("Code-Review", ownerGroupId, devGroupId));permissions.put("forgeAuthor", buildPermissionInfo("", devGroupId));permissions.put("editTopicName", buildPermissionInfo("", devGroupId));permissions.put("create", buildPermissionInfo("", devGroupId));permissions.put("push", buildPermissionInfo("", devGroupId));permissions.put("submit", buildPermissionInfo("", ownerGroupId));accessSectionInfo.permissions = permissions;return accessSectionInfo;}private PermissionInfo buildPermissionInfo(String label, String groupId) {PermissionInfo permissionInfo = new PermissionInfo(label, false);Map<String, PermissionRuleInfo> rules = buildRules(groupId);permissionInfo.rules = rules;return permissionInfo;}private PermissionInfo buildPermissionInfo(String label, String ownerGroupId, String devGroupId) {PermissionInfo permissionInfo = new PermissionInfo(label, false);Map<String, PermissionRuleInfo> rules = buildRules(ownerGroupId, devGroupId);permissionInfo.rules = rules;return permissionInfo;}private PermissionInfo buildReviewPermissionInfo(String label, String ownerGroupId, String devGroupId) {PermissionInfo permissionInfo = new PermissionInfo(label, false);Map<String, PermissionRuleInfo> rules = buildReviewRules(ownerGroupId, devGroupId);permissionInfo.rules = rules;return permissionInfo;}private Map<String, PermissionRuleInfo> buildRules(String groupId) {Map<String, PermissionRuleInfo> rules = Maps.newHashMap();PermissionRuleInfo permissionRuleInfo = buildPermissionRuleInfo();rules.put(groupId, permissionRuleInfo);return rules;}private Map<String, PermissionRuleInfo> buildRules(String ownerGroupId, String devGroupId) {Map<String, PermissionRuleInfo> rules = Maps.newHashMap();PermissionRuleInfo permissionRuleInfo = buildPermissionRuleInfo();rules.put(ownerGroupId, permissionRuleInfo);rules.put(devGroupId, permissionRuleInfo);return rules;}private Map<String, PermissionRuleInfo> buildReviewRules(String ownerGroupId, String devGroupId) {Map<String, PermissionRuleInfo> rules = Maps.newHashMap();PermissionRuleInfo permissionRuleInfo = buildPermissionRuleInfo();permissionRuleInfo.max = 2;permissionRuleInfo.min = -2;rules.put(ownerGroupId, permissionRuleInfo);permissionRuleInfo = buildPermissionRuleInfo();permissionRuleInfo.max = 1;permissionRuleInfo.min = -1;rules.put(devGroupId, permissionRuleInfo);return rules;}private PermissionRuleInfo buildPermissionRuleInfo() {PermissionRuleInfo permissionRuleInfo = new PermissionRuleInfo(PermissionRuleInfo.Action.ALLOW, false);return permissionRuleInfo;}

三、测试

    @Autowiredprivate GerritService gerritService;@ApiOperation(value = "新建gerrit工程")@PostMapping(value = "/api/gerrit/create/{projectName}")public ResponseEntity<?> createProject(@NotNull @PathVariable String projectName) {// 新建gerrit工程gerritService.create("", "root/" + projectName);return ResponseEntity.ok("SUCCESS");}

四、写在最后的话

截止到目前,对接gerrit的设计与详细实现均已完整梳理,后文将解决代码提交的消息提醒。

本文只实现了自动创建gerrit项目以及权限项,创建了owner组和dev组。因为缺少信息–每个组里应该添加哪些人。
后期将实现给组里加人的操作。

也就是说,gerrit工程创建后,申请新项目的人,如果不在上面的owner组内,你会看不到已创建好的gerrti工程!!

不要慌,让owner组里的人员把你加入进组里。

相关文章:

Devops系列之对接Gerrit的设计与实现(三)-- Java编程实现

一、背景 上文讲述了如何使用shell命令实现创建gerrit项目&#xff0c;本文介绍如何使用java语言编程实现。 二、java语言实现 1、引入jar包 <dependency><groupId>com.urswolfer.gerrit.client.rest</groupId><artifactId>gerrit-rest-java-client…...

深入理解全排列算法:DFS与回溯的完美结合

全排列问题是算法中的经典问题&#xff0c;其目标是将一组数字的所有可能排列组合列举出来。本文将详细解析如何通过深度优先搜索&#xff08;DFS&#xff09;和回溯法高效生成全排列&#xff0c;并通过模拟递归过程帮助读者彻底掌握其核心思想。 问题描述 给定一个正整数 n&a…...

服务器(一种管理计算资源的计算机)

服务器是在网络环境中提供计算能力并运行软件应用程序的特定IT设备&#xff0c;它在网络中为其他客户机&#xff08;如个人计算机、智能手机、ATM机等终端设备&#xff09;提供计算或者应用服务, 一般来说服务器都具备承担响应服务请求、承担服务、保障服务的能力。服务器相比普…...

时态--02--一般过去时

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一般过去时1.肯定句am/is — wasare — were 2.否定句3.⼀般疑问句4.特殊疑问句5.there be 过去式 practice过去分词 一般过去时 1.肯定句 am/is — was are — wer…...

WSA(Windows Subsystem for Android)安装LSPosed和应用教程

windows安卓子系统WSA的Lsposed和shamiko的安装教程 WSA(Windows Subsystem for Android)安装LSPosed和应用教程 一、环境准备 在开始之前,请确保: 已经安装好WSA(Windows Subsystem for Android)已经安装好ADB工具下载好LSPosed和Shamiko框架安装包 二、连接WSA 首先需要…...

Opencv计算机视觉编程攻略-第十三节 跟踪视频中的物品

这是opencv系列的最后一节&#xff0c;主要学习视频序列&#xff0c;上一节介绍了读取、处理和存储视频的工具&#xff0c;本文将介绍几种跟踪图像序列中运动物体的算法。可见运动或表观运动&#xff0c;是物体以不同的速度在不同的方向上移动&#xff0c;或者是因为相机在移动…...

10 个最新 CSS 功能已在所有主流浏览器中得到支持

前言 CSS 不断发展&#xff0c;新功能使我们的工作更快、更简洁、更强大。得益于最新的浏览器改进&#xff08;Baseline 2024&#xff09;&#xff0c;许多新功能现在可在所有主要引擎上使用。以下是您可以立即开始使用的10 CSS新功能。 1. Scrollbar-Gutter 和 Scrollbar-Co…...

[特殊字符] 企业级Docker私有仓库实战:3步搭建Harbor安全仓库,镜像管理从此高效无忧

本文提供 一站式Docker私有仓库部署指南&#xff0c;聚焦企业级镜像管理需求&#xff0c;深入解析Harbor私有仓库的搭建、运维与安全加固全流程。内容涵盖 轻量级Registry快速部署与 Harbor企业级方案对比&#xff0c;手把手演示SSL证书配置、多租户权限控制、镜像漏洞扫描等核…...

一个基于Django的进销存管理系统Demo实现

第一步&#xff1a;创建 Django 项目 bash 复制 django-admin startproject inventory_system cd inventory_system python manage.py startapp erp 第二步&#xff1a;定义数据模型&#xff08;models.py&#xff09; python 复制 from django.db import models from d…...

wsl2+ubuntu22.04安装blender教程(详细教程)

本章教程介绍,如何在Windows操作系统上通过wsl2+ubuntu安装blender并运行教程。Blender 是一款免费、开源的 ​​3D 创作套件​​,广泛应用于建模、动画、渲染、视频编辑、特效制作等领域。它由全球开发者社区共同维护,支持跨平台(Windows、macOS、Linux),功能强大且完全…...

netty中的ChannelPipeline详解

Netty中的ChannelPipeline是事件处理链的核心组件,负责将多个ChannelHandler组织成有序的责任链,实现网络事件(如数据读写、连接状态变化)的动态编排和传播。以下从核心机制、执行逻辑到应用场景进行详细解析: 1. 核心结构与组成 双向链表结构 组成单元:ChannelPipeline…...

使用多进程和 Socket 接收解析数据并推送到 Kafka 的高性能架构

使用多进程和 Socket 接收解析数据并推送到 Kafka 的高性能架构 在现代应用程序中&#xff0c;实时数据处理和高并发性能是至关重要的。本文将介绍如何使用 Python 的多进程和 Socket 技术来接收和解析数据&#xff0c;并将处理后的数据推送到 Kafka&#xff0c;从而实现高效的…...

WinForm真入门(14)——ListView控件详解

一、ListView 控件核心概念与功能 ‌ListView‌ 是 WinForm 中用于展示结构化数据的多功能列表控件&#xff0c;支持多列、多视图模式及复杂交互&#xff0c;常用于文件资源管理器、数据报表等场景‌。 核心特点‌&#xff1a; 支持 ‌5种视图模式‌&#xff1a;Details&…...

FastAPI用户认证系统开发指南:从零构建安全API

前言 在现代Web应用开发中&#xff0c;用户认证系统是必不可少的功能。本文将带你使用FastAPI框架构建一个完整的用户认证系统&#xff0c;包含注册、登录、信息更新和删除等功能。我们将采用JWT&#xff08;JSON Web Token&#xff09;进行身份验证&#xff0c;并使用SQLite作…...

【BUG】阿里云服务器数据库远程连接报错

当你遇到 ERROR 2003 (HY000): Cant connect to MySQL server on 47.100.xxx.xx (10061) 错误&#xff0c;这个错误代码 10061 通常意味着客户端无法连接到指定的 MySQL 服务器&#xff0c;原因可能有多种&#xff0c;下面为你分析可能的原因及对应的解决办法。 1. 网络连接问…...

【前端】【React】性能优化三件套useCallback,useMemo,React.memo

一、总览&#xff1a;性能优化三件套 useCallback(fn, deps)&#xff1a;缓存函数&#xff0c;避免每次渲染都新建函数。useMemo(fn, deps)&#xff1a;缓存值&#xff08;计算结果&#xff09;&#xff0c;避免重复执行计算。React.memo(Component)&#xff1a;缓存组件的渲染…...

Vue3性能优化终极指南:编译策略、运行时调优与全链路监控

一、Vue3性能优化体系框架 1.1 性能优化全景图谱 1.2 关键性能指标定义表 指标测量方式优化目标核心影响因子FCPLighthouse<1.5s资源加载速度LCPPerformance API<2.5s关键资源大小TTIWebPageTest<3.5s主线程阻塞时间Memory UsageChrome DevTools<50MB对象引用策略…...

FISCO BCOS技术架构解析:从多群组设计到性能优化实践

目录 FISCO BCOS整体架构设计 多群组架构与数据隔离机制 交易流程与执行机制 安全架构与隐私保护 性能优化与压测实践 应用案例与生态工具 FISCO BCOS作为中国领先的金融级开源联盟链平台,自2017年由金链盟开源工作组推出以来,已在政务、金融、医疗、版权等众多领域实现…...

Ceph异地数据同步之- S3对象异地同步复制

#作者&#xff1a;闫乾苓 文章目录 关键组件说明数据流说明部署步骤配置主区域配置次要区域S3对象文件同步测试 关键组件说明 在Ceph RGW的多站点复制架构中&#xff0c;Realm、Zonegroup 和 Zone 是关键的组织结构&#xff0c;用于管理多站点的配置和数据同步 Realm(领域)&a…...

iOS按键精灵辅助工具在游戏开发中的创新应用

一、iOS自动化测试辅助工具 在移动游戏开发领域&#xff0c;iOS按键精灵类辅助工具不同于传统的安卓自动化方案&#xff0c;iOS环境下的自动化测试面临更严峻的技术挑战&#xff0c;但通过创新方法仍可实现精准控制。 # 基于图像识别的智能定位算法示例 def find_button(butt…...

3D案例丨多个3D工业相机拼接检测 开启360°新视界

在高速生产线上&#xff0c;经常需要在极短的时间内对工件进行全方位的外观检测&#xff0c;如&#xff1a;线缆直径和直线度检测、锂电池外观缺陷检测、铁轨截面尺寸检测等。 这需要传感器完整还原被测物的截面面轮廓形状&#xff0c;并获取精准的截面轮廓数据。但单一相机的…...

打分函数分类

在分子对接中&#xff0c;打分函数用于评估配体与受体结合的亲和力。不同类型的打分函数有各自的优势和应用场景。常见的打分函数主要分为以下几类&#xff1a; 1. 基于物理&#xff08;力场&#xff09;的打分函数 (Force/physics-field-based scoring functions) 这种打分…...

实践 DevOps 项目:使用 Terraform、Helm、SonarQube 和 GitLab CI/CD 在 AWS EKS 上实践全栈部署

在当今快节奏的软件开发领域&#xff0c;自动化至关重要。在本文中&#xff0c;我将向您展示如何构建一个全面的 DevOps 流水线&#xff0c;该流水线能够&#xff1a; 使用 Terraform 预置完整的 AWS 基础设施。部署一个包含私有子网和公共子网、RDS PostgreSQL 以及完整配置的…...

EFT干扰和共模干扰

EFT干扰本质上属于共模干扰的一种具体表现形式&#xff0c;但严格来说不能简单等同于共模干扰。以下从原理、特征及区别角度展开分析&#xff1a; 1. EFT干扰的原理 定义&#xff1a;EFT&#xff08;Electrical Fast Transient&#xff0c;电快速瞬变脉冲群&#xff09;干扰是…...

android 下提示 SQLITECIPHER driver not loaded

问题描述: 在android下出现 SQLITECIPHER driver not loaded 错误 解决办法: 在QT的Android目录下面放入 libplugins_sqldrivers_sqlitecipher_arm64-v8a.so...

[D1,2]回溯刷题

文章目录 组合 组合 回溯的基础结构 #组合总和 注意startIndex的更新是用i来更新的&#xff0c;不然会产生重复的组合...

使用 VBA 宏创建一个选择全部word图片快捷指令,进行图片格式编辑

使用 VBA 宏批量选择图片 ✅ 第一步&#xff1a;创建 .dotm 加载项文件 1、使用环境 office word 365&#xff0c;文件格式为.docx 图片格式为.PNG 2、创建 .dotm 加载项文件 打开 Word&#xff0c;新建一个空白文档。 按下 Alt F11 打开 VBA 编辑器。 点击菜单栏&#xff…...

SQL 关键字

SQL 包含许多关键字&#xff0c;这些关键字用于执行各种数据库操作。以下是主要的 SQL 关键字分类&#xff1a; 数据查询语言 (DQL) SELECT - 从数据库中选择数据 FROM - 指定要查询的表 WHERE - 指定查询条件 GROUP BY - 对结果集进行分组 HAVING - 对分组结果进行过滤 …...

从PPT到PNG:Python实现的高效PPT转图工具

从PPT到PNG&#xff1a;Python实现的高效PPT转图工具 在日常工作中&#xff0c;PPT&#xff08;PowerPoint&#xff09;文件是我们常用的演示工具。然而&#xff0c;有时候我们需要将PPT的内容提取为图片格式&#xff08;如PNG&#xff09;以便于展示或保存。手动将每一页PPT保…...

TCP和UDP协议

前言 TCP&#xff08;传输控制协议&#xff09;和UDP&#xff08;用户数据报协议&#xff09;是两种主要的传输层协议&#xff1b;它们在连接方式、可靠性、效率等方面有显著区别。 关键对比 差异总结 可靠性&#xff1a; TCP通过确认应答、重传等机制确保数据可靠传输&#…...

高并发内存池(三):PageCache(页缓存)的实现

前言&#xff1a; 在前两期内容中&#xff0c;我们深入探讨了内存管理机制中在 ThreadCache 和 CentralCache两个层级进行内存申请的具体实现。这两层缓存作为高效的内存分配策略&#xff0c;能够快速响应线程的内存需求&#xff0c;减少锁竞争&#xff0c;提升程序性能。 本期…...

使用pybind11开发可供python使用的c++扩展模块

在做紫微斗数程序的时候用到了padas库,不过也只用了它下面几个功能: 1、读入csv文件,构造DataFrame; 2、通过行列标题查找数据; 3、通过行标题读取一行数据。 用这几个功能却导入了pandas、numpy、dateutil、pytz等一堆库,多少有点划不来,于是想用c++开发一个实现这几…...

系统与网络安全------网络通信原理(5)

资料整理于网络资料、书本资料、AI&#xff0c;仅供个人学习参考。 传输层解析 传输层 传输层的作用 IP层提供点到点的连接传输层提供端到端的连接 端口到端口的连接&#xff08;不同端口号&#xff0c;代表不同的应用程序&#xff09; TCP协议概述 TCP&#xff08;Transm…...

JavaScript防抖与节流

目录 防抖&#xff08;Debounce&#xff09; 一、防抖的定义 二、防抖的实现原理 三、防抖的代码实现 四、代码解析 五、使用示例 1. 输入框实时搜索&#xff08;延迟执行模式&#xff09; 2. 按钮防重复点击&#xff08;立即执行模式&#xff09; 六、总结 节流&…...

Java网络编程实战(多人聊天室-CS模式)

一、C/S模式核心原理 1.1 基本架构 C/S&#xff08;Client/Server&#xff09;模式采用客户端-服务器架构&#xff1a; 服务器端&#xff1a;持续运行&#xff0c;负责消息路由和广播客户端&#xff1a;用户交互界面&#xff0c;连接服务器进行通信通信协议&#xff1a;TCP&…...

Vue3.5 + Vite6.x 项目的完整 Stylelint 配置方案,支持 .vue/.html 内联样式、Less/SCSS/CSS 等多种文件类

Vue3.5 Vite6.x 项目的完整 Stylelint 配置方案&#xff0c;支持 .vue/.html 内联样式、Less/SCSS/CSS 等多种文件类型 一、完整依赖安装 npm install --save-dev stylelint stylelint-config-standard postcss-html # 解析 Vue/HTML 文件中的样式postcss-scss …...

23种设计模式Java版(带脑图,带示例源码)

设计模式 1、创建型 1.1、单例模式(Singleton pattern) 确保一个类只有一个实例&#xff0c;并提供该实例的全局访问点。 1.2、工厂方法(Factory Method) 它定义了一个创建对象的接口&#xff0c;但由子类决定要实例化哪个类。工厂方法把实例化操作推迟到子类。 1.3、抽象…...

mapbox高阶,使用graphology、graphology-shortest-path前端插件和本地geojson数据纯前端实现路径规划

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言1.1 ☘️mapboxgl.Map 地图对象1.2 ☘️mapboxgl.Map style属性1.3 ☘️graphology 插件1.3.1 ☘️概念1.3.2 ☘…...

【已解决】vscode升级后连接远程异常:“远程主机可能不符合XXX的先决条件”解决方法

vscode提示升级&#xff0c;每次都升了&#xff0c;突然某次关闭后无法连接远程&#xff0c;查询资料是因为从VS Code 1.86.1版本开始&#xff08;2024年1月&#xff09;要求glibc版本>2.28。 命令“ ldd --version”可查看glibc版本为2.27&#xff1a; rootXXXXXXX:~$ ld…...

Springboot整合JAVAFX

Springboot整合JAVAFX 实体与VO设计 pom.xml文件如下&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xs…...

【算法】——一键解决动态规划

前言 动态规划是一种高效解决​​重叠子问题​​和​​最优子结构​​问题的算法思想。它通过​​分治记忆化​​&#xff0c;将复杂问题分解为子问题&#xff0c;并存储中间结果&#xff0c;避免重复计算&#xff0c;从而大幅提升效率。 ​​为什么重要&#xff1f;​ ​​优化…...

Git使用与管理

一.基本操作 1.创建本地仓库 在对应文件目录下进行&#xff1a; git init 输入完上面的代码&#xff0c;所在文件目录下就会多一个名为 .git 的隐藏文件&#xff0c;该文件是Git用来跟踪和管理仓库的。 我们可以使用 tree 命令&#xff08;注意要先下载tree插件&#xff09…...

npm、nvm、nrm

NVM (Node Version Manager) 常见指令 NVM 是一个用于管理 Node.js 版本的流行工具&#xff0c;允许你在同一台机器上安装和切换不同版本的 Node.js。以下是 NVM 的常见指令&#xff1a; 安装与卸载 nvm install <version> - 安装指定版本的 Node.js 例如&#xff1a;…...

Java 文件内容转换为MD5哈希值

若要把读取到的 files 列表里的内容转换为 MD5 哈希值&#xff0c;你可以逐个遍历 files 列表中的元素&#xff0c;将每个元素的内容计算成 MD5 哈希值。 以下是一个完整的 Java 示例代码&#xff0c;展示了如何实现这一功能&#xff1a; import java.io.BufferedInputStream…...

未来郴州:科技与自然的交响诗篇

故事背景 故事发生在中国湖南郴州&#xff0c;描绘了未来城市中科技与自然共生共荣的奇妙图景。通过六个充满诗意的场景&#xff0c;展现雾能转化系统、立体生态书库、智能稻田等创新设计&#xff0c;编织出一曲人类智慧与自然韵律共鸣的未来交响。 故事内容 在东江湖的晨雾中&…...

UE5 运行时动态将玩家手部模型设置为相机的子物体

在编辑器里&#xff0c;我们虽然可以手动添加相机&#xff0c;但是无法将网格体设置为相机的子物体&#xff0c;只能将相机设置为网格体的子物体 但是为了使用方便&#xff0c;我们希望将网格体设置为相机的子物体&#xff0c;这样我们直接旋转相机就可以旋转网格体&#xff0…...

Ubuntu系统下的包管理器APT

Ubuntu系统下的包管理器APT 在Linux操作系统生态中&#xff0c;软件包管理工具是连接用户与系统功能的桥梁。Ubuntu作为基于Debian的流行发行版&#xff0c;其强大的包管理系统APT&#xff08;Advanced Packaging Tool&#xff09;为开发者与系统管理员提供了便捷的软件生命周…...

超级码科技发布镂空AI保险胶带,重塑包装防伪新标准

在酒类、物流、奢侈品、电子产品等领域&#xff0c;包装安全与防伪需求日益迫切。传统封箱胶带易被转移或重复利用&#xff0c;导致商品被仿冒的风险居高不下。 为此&#xff0c;超级码科技推出镂空型防揭AI数字身份保险封箱胶带——一款集结构防伪、信息追踪与增值服务于一体的…...

微软Exchange管理中心全球范围宕机

微软已确认Exchange管理中心&#xff08;Exchange Admin Center&#xff0c;EAC&#xff09;发生全球性服务中断&#xff0c;导致管理员无法访问关键管理工具。该故障被标记为关键服务事件&#xff08;编号EX1051697&#xff09;&#xff0c;对依赖Exchange Online的企业造成广…...

前端通信库fetch-event-source实现丰富的SSE

环境:SpringBoot3.4.0 + Vue3 1. 简介 SSE(Server-Sent Events)是一种基于HTTP的服务器向客户端单向推送实时数据的轻量级协议,配合浏览器原生EventSource API,可实现高效实时通信。前端通过创建EventSource对象订阅服务端流,自动处理连接、重试与数据解析;服务端设置C…...