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项目,本文介绍如何使用java语言编程实现。 二、java语言实现 1、引入jar包 <dependency><groupId>com.urswolfer.gerrit.client.rest</groupId><artifactId>gerrit-rest-java-client…...
深入理解全排列算法:DFS与回溯的完美结合
全排列问题是算法中的经典问题,其目标是将一组数字的所有可能排列组合列举出来。本文将详细解析如何通过深度优先搜索(DFS)和回溯法高效生成全排列,并通过模拟递归过程帮助读者彻底掌握其核心思想。 问题描述 给定一个正整数 n&a…...
服务器(一种管理计算资源的计算机)
服务器是在网络环境中提供计算能力并运行软件应用程序的特定IT设备,它在网络中为其他客户机(如个人计算机、智能手机、ATM机等终端设备)提供计算或者应用服务, 一般来说服务器都具备承担响应服务请求、承担服务、保障服务的能力。服务器相比普…...
时态--02--一般过去时
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 一般过去时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系列的最后一节,主要学习视频序列,上一节介绍了读取、处理和存储视频的工具,本文将介绍几种跟踪图像序列中运动物体的算法。可见运动或表观运动,是物体以不同的速度在不同的方向上移动,或者是因为相机在移动…...
10 个最新 CSS 功能已在所有主流浏览器中得到支持
前言 CSS 不断发展,新功能使我们的工作更快、更简洁、更强大。得益于最新的浏览器改进(Baseline 2024),许多新功能现在可在所有主要引擎上使用。以下是您可以立即开始使用的10 CSS新功能。 1. Scrollbar-Gutter 和 Scrollbar-Co…...
[特殊字符] 企业级Docker私有仓库实战:3步搭建Harbor安全仓库,镜像管理从此高效无忧
本文提供 一站式Docker私有仓库部署指南,聚焦企业级镜像管理需求,深入解析Harbor私有仓库的搭建、运维与安全加固全流程。内容涵盖 轻量级Registry快速部署与 Harbor企业级方案对比,手把手演示SSL证书配置、多租户权限控制、镜像漏洞扫描等核…...
一个基于Django的进销存管理系统Demo实现
第一步:创建 Django 项目 bash 复制 django-admin startproject inventory_system cd inventory_system python manage.py startapp erp 第二步:定义数据模型(models.py) 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 的高性能架构 在现代应用程序中,实时数据处理和高并发性能是至关重要的。本文将介绍如何使用 Python 的多进程和 Socket 技术来接收和解析数据,并将处理后的数据推送到 Kafka,从而实现高效的…...
WinForm真入门(14)——ListView控件详解
一、ListView 控件核心概念与功能 ListView 是 WinForm 中用于展示结构化数据的多功能列表控件,支持多列、多视图模式及复杂交互,常用于文件资源管理器、数据报表等场景。 核心特点: 支持 5种视图模式:Details&…...
FastAPI用户认证系统开发指南:从零构建安全API
前言 在现代Web应用开发中,用户认证系统是必不可少的功能。本文将带你使用FastAPI框架构建一个完整的用户认证系统,包含注册、登录、信息更新和删除等功能。我们将采用JWT(JSON Web Token)进行身份验证,并使用SQLite作…...
【BUG】阿里云服务器数据库远程连接报错
当你遇到 ERROR 2003 (HY000): Cant connect to MySQL server on 47.100.xxx.xx (10061) 错误,这个错误代码 10061 通常意味着客户端无法连接到指定的 MySQL 服务器,原因可能有多种,下面为你分析可能的原因及对应的解决办法。 1. 网络连接问…...
【前端】【React】性能优化三件套useCallback,useMemo,React.memo
一、总览:性能优化三件套 useCallback(fn, deps):缓存函数,避免每次渲染都新建函数。useMemo(fn, deps):缓存值(计算结果),避免重复执行计算。React.memo(Component):缓存组件的渲染…...
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对象异地同步复制
#作者:闫乾苓 文章目录 关键组件说明数据流说明部署步骤配置主区域配置次要区域S3对象文件同步测试 关键组件说明 在Ceph RGW的多站点复制架构中,Realm、Zonegroup 和 Zone 是关键的组织结构,用于管理多站点的配置和数据同步 Realm(领域)&a…...
iOS按键精灵辅助工具在游戏开发中的创新应用
一、iOS自动化测试辅助工具 在移动游戏开发领域,iOS按键精灵类辅助工具不同于传统的安卓自动化方案,iOS环境下的自动化测试面临更严峻的技术挑战,但通过创新方法仍可实现精准控制。 # 基于图像识别的智能定位算法示例 def find_button(butt…...
3D案例丨多个3D工业相机拼接检测 开启360°新视界
在高速生产线上,经常需要在极短的时间内对工件进行全方位的外观检测,如:线缆直径和直线度检测、锂电池外观缺陷检测、铁轨截面尺寸检测等。 这需要传感器完整还原被测物的截面面轮廓形状,并获取精准的截面轮廓数据。但单一相机的…...
打分函数分类
在分子对接中,打分函数用于评估配体与受体结合的亲和力。不同类型的打分函数有各自的优势和应用场景。常见的打分函数主要分为以下几类: 1. 基于物理(力场)的打分函数 (Force/physics-field-based scoring functions) 这种打分…...
实践 DevOps 项目:使用 Terraform、Helm、SonarQube 和 GitLab CI/CD 在 AWS EKS 上实践全栈部署
在当今快节奏的软件开发领域,自动化至关重要。在本文中,我将向您展示如何构建一个全面的 DevOps 流水线,该流水线能够: 使用 Terraform 预置完整的 AWS 基础设施。部署一个包含私有子网和公共子网、RDS PostgreSQL 以及完整配置的…...
EFT干扰和共模干扰
EFT干扰本质上属于共模干扰的一种具体表现形式,但严格来说不能简单等同于共模干扰。以下从原理、特征及区别角度展开分析: 1. EFT干扰的原理 定义:EFT(Electrical Fast Transient,电快速瞬变脉冲群)干扰是…...
android 下提示 SQLITECIPHER driver not loaded
问题描述: 在android下出现 SQLITECIPHER driver not loaded 错误 解决办法: 在QT的Android目录下面放入 libplugins_sqldrivers_sqlitecipher_arm64-v8a.so...
[D1,2]回溯刷题
文章目录 组合 组合 回溯的基础结构 #组合总和 注意startIndex的更新是用i来更新的,不然会产生重复的组合...
使用 VBA 宏创建一个选择全部word图片快捷指令,进行图片格式编辑
使用 VBA 宏批量选择图片 ✅ 第一步:创建 .dotm 加载项文件 1、使用环境 office word 365,文件格式为.docx 图片格式为.PNG 2、创建 .dotm 加载项文件 打开 Word,新建一个空白文档。 按下 Alt F11 打开 VBA 编辑器。 点击菜单栏ÿ…...
SQL 关键字
SQL 包含许多关键字,这些关键字用于执行各种数据库操作。以下是主要的 SQL 关键字分类: 数据查询语言 (DQL) SELECT - 从数据库中选择数据 FROM - 指定要查询的表 WHERE - 指定查询条件 GROUP BY - 对结果集进行分组 HAVING - 对分组结果进行过滤 …...
从PPT到PNG:Python实现的高效PPT转图工具
从PPT到PNG:Python实现的高效PPT转图工具 在日常工作中,PPT(PowerPoint)文件是我们常用的演示工具。然而,有时候我们需要将PPT的内容提取为图片格式(如PNG)以便于展示或保存。手动将每一页PPT保…...
TCP和UDP协议
前言 TCP(传输控制协议)和UDP(用户数据报协议)是两种主要的传输层协议;它们在连接方式、可靠性、效率等方面有显著区别。 关键对比 差异总结 可靠性: TCP通过确认应答、重传等机制确保数据可靠传输&#…...
高并发内存池(三):PageCache(页缓存)的实现
前言: 在前两期内容中,我们深入探讨了内存管理机制中在 ThreadCache 和 CentralCache两个层级进行内存申请的具体实现。这两层缓存作为高效的内存分配策略,能够快速响应线程的内存需求,减少锁竞争,提升程序性能。 本期…...
使用pybind11开发可供python使用的c++扩展模块
在做紫微斗数程序的时候用到了padas库,不过也只用了它下面几个功能: 1、读入csv文件,构造DataFrame; 2、通过行列标题查找数据; 3、通过行标题读取一行数据。 用这几个功能却导入了pandas、numpy、dateutil、pytz等一堆库,多少有点划不来,于是想用c++开发一个实现这几…...
系统与网络安全------网络通信原理(5)
资料整理于网络资料、书本资料、AI,仅供个人学习参考。 传输层解析 传输层 传输层的作用 IP层提供点到点的连接传输层提供端到端的连接 端口到端口的连接(不同端口号,代表不同的应用程序) TCP协议概述 TCP(Transm…...
JavaScript防抖与节流
目录 防抖(Debounce) 一、防抖的定义 二、防抖的实现原理 三、防抖的代码实现 四、代码解析 五、使用示例 1. 输入框实时搜索(延迟执行模式) 2. 按钮防重复点击(立即执行模式) 六、总结 节流&…...
Java网络编程实战(多人聊天室-CS模式)
一、C/S模式核心原理 1.1 基本架构 C/S(Client/Server)模式采用客户端-服务器架构: 服务器端:持续运行,负责消息路由和广播客户端:用户交互界面,连接服务器进行通信通信协议:TCP&…...
Vue3.5 + Vite6.x 项目的完整 Stylelint 配置方案,支持 .vue/.html 内联样式、Less/SCSS/CSS 等多种文件类
Vue3.5 Vite6.x 项目的完整 Stylelint 配置方案,支持 .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) 确保一个类只有一个实例,并提供该实例的全局访问点。 1.2、工厂方法(Factory Method) 它定义了一个创建对象的接口,但由子类决定要实例化哪个类。工厂方法把实例化操作推迟到子类。 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提示升级,每次都升了,突然某次关闭后无法连接远程,查询资料是因为从VS Code 1.86.1版本开始(2024年1月)要求glibc版本>2.28。 命令“ ldd --version”可查看glibc版本为2.27: rootXXXXXXX:~$ ld…...
Springboot整合JAVAFX
Springboot整合JAVAFX 实体与VO设计 pom.xml文件如下: <?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…...
【算法】——一键解决动态规划
前言 动态规划是一种高效解决重叠子问题和最优子结构问题的算法思想。它通过分治记忆化,将复杂问题分解为子问题,并存储中间结果,避免重复计算,从而大幅提升效率。 为什么重要? 优化…...
Git使用与管理
一.基本操作 1.创建本地仓库 在对应文件目录下进行: git init 输入完上面的代码,所在文件目录下就会多一个名为 .git 的隐藏文件,该文件是Git用来跟踪和管理仓库的。 我们可以使用 tree 命令(注意要先下载tree插件)…...
npm、nvm、nrm
NVM (Node Version Manager) 常见指令 NVM 是一个用于管理 Node.js 版本的流行工具,允许你在同一台机器上安装和切换不同版本的 Node.js。以下是 NVM 的常见指令: 安装与卸载 nvm install <version> - 安装指定版本的 Node.js 例如:…...
Java 文件内容转换为MD5哈希值
若要把读取到的 files 列表里的内容转换为 MD5 哈希值,你可以逐个遍历 files 列表中的元素,将每个元素的内容计算成 MD5 哈希值。 以下是一个完整的 Java 示例代码,展示了如何实现这一功能: import java.io.BufferedInputStream…...
未来郴州:科技与自然的交响诗篇
故事背景 故事发生在中国湖南郴州,描绘了未来城市中科技与自然共生共荣的奇妙图景。通过六个充满诗意的场景,展现雾能转化系统、立体生态书库、智能稻田等创新设计,编织出一曲人类智慧与自然韵律共鸣的未来交响。 故事内容 在东江湖的晨雾中&…...
UE5 运行时动态将玩家手部模型设置为相机的子物体
在编辑器里,我们虽然可以手动添加相机,但是无法将网格体设置为相机的子物体,只能将相机设置为网格体的子物体 但是为了使用方便,我们希望将网格体设置为相机的子物体,这样我们直接旋转相机就可以旋转网格体࿰…...
Ubuntu系统下的包管理器APT
Ubuntu系统下的包管理器APT 在Linux操作系统生态中,软件包管理工具是连接用户与系统功能的桥梁。Ubuntu作为基于Debian的流行发行版,其强大的包管理系统APT(Advanced Packaging Tool)为开发者与系统管理员提供了便捷的软件生命周…...
超级码科技发布镂空AI保险胶带,重塑包装防伪新标准
在酒类、物流、奢侈品、电子产品等领域,包装安全与防伪需求日益迫切。传统封箱胶带易被转移或重复利用,导致商品被仿冒的风险居高不下。 为此,超级码科技推出镂空型防揭AI数字身份保险封箱胶带——一款集结构防伪、信息追踪与增值服务于一体的…...
微软Exchange管理中心全球范围宕机
微软已确认Exchange管理中心(Exchange Admin Center,EAC)发生全球性服务中断,导致管理员无法访问关键管理工具。该故障被标记为关键服务事件(编号EX1051697),对依赖Exchange Online的企业造成广…...
前端通信库fetch-event-source实现丰富的SSE
环境:SpringBoot3.4.0 + Vue3 1. 简介 SSE(Server-Sent Events)是一种基于HTTP的服务器向客户端单向推送实时数据的轻量级协议,配合浏览器原生EventSource API,可实现高效实时通信。前端通过创建EventSource对象订阅服务端流,自动处理连接、重试与数据解析;服务端设置C…...