开发者的测试复盘:架构分层测试策略与工具链闭环设计实战
摘要
针对测试复盘流于形式、覆盖率虚高等行业痛点,本文提出一套结合架构分层与工具链闭环的解决方案:
- 分层测试策略精准化:通过单元测试精准狙击核心逻辑、契约测试驱动接口稳定性、黄金链路固化端到端场景,实现缺陷拦截率提升;
- 工具链自动化闭环:基于Spring Cloud Contract实现消费者驱动的契约验证,结合Testcontainers构建轻量化环境治理体系;
- 团队协作范式升级:从“被动救火”到“测试左移”,通过需求阶段验收条件绑定与代码提交卡点,降低缺陷逃逸率。
一、为什么测试复盘沦为“走过场”?
许多团队的测试复盘文档常沦为以下模板:
1. 发现问题:接口超时、数据不一致
2. 解决方案:优化SQL、增加缓存
3. 后续计划:加强监控
弊端:缺乏对测试体系本身的反思,无法形成持续改进机制。
高质量复盘的核心目标:
- 暴露流程缺陷:如单元测试缺失导致低级Bug频发
- 验证工具有效性:如自动化测试是否覆盖关键路径
- 量化效能提升:如回归测试耗时从2小时→15分钟
二、架构分层视角下的测试策略设计
1. 单元测试:从“满足覆盖率”到“关键逻辑验证”
目标:验证代码逻辑的最小单元(如类、方法)的正确性。
误区:盲目追求行覆盖率(Line Coverage),忽视复杂分支验证。
案例:订单服务优惠券计算逻辑:
// 原始测试:仅覆盖满100减10的基础场景
@Test
public void testCouponCalculation() { double result = calculator.apply(100, "FIXED_10"); assertEquals(90, result);
} // 优化后:覆盖叠加优惠、过期券异常等边界
@Test
public void testOverlappingCoupons() { // 组合优惠券叠加逻辑验证
}
@Test(expected = CouponExpiredException.class)
public void testExpiredCoupon() { // 过期优惠券触发异常
}
成果:单元测试缺陷拦截率提升40%,复杂场景覆盖率从55%→92%。
2. 集成测试:用契约测试替代“脆弱的Mock”
目标:验证模块间交互(如API调用、数据库访问)。
痛点:传统Mock导致测试与真实环境脱节,接口变更易引发误报。
解决方案:
- 消费者驱动的契约测试(CDC):
- 消费者定义接口预期(Spring Cloud Contract);
- 提供方自动生成验证用例并绑定Swagger文档;
// 支付服务契约(消费者端)
Contract.make { request { method POST() url "/api/payments" body([orderId: "123", amount: 199.0]) } response { status 201 body([paymentId: "pay_2023", status: "SUCCESS"]) }
}
- Testcontainers替代本地Mock:在Docker容器中启动真实MySQL、Redis依赖
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
@Testcontainers
public class OrderServiceIntegrationTest {@Autowiredprivate TestRestTemplate restTemplate; // 注入HTTP测试客户端@Autowiredprivate DataSource dataSource;@Autowiredprivate StringRedisTemplate redisTemplate;@Containerstatic final MySQLContainer<?> mysql = new MySQLContainer<>("mysql:8.0.32.1").withDatabaseName("orders_db").withUsername("admin").withPassword("admin123");@Containerstatic final GenericContainer<?> redis = new GenericContainer<>("redis:7.0.11.1").withExposedPorts(6379);@DynamicPropertySourcestatic void registerProperties(DynamicPropertyRegistry registry) {// 配置MySQLregistry.add("spring.datasource.url", mysql::getJdbcUrl);registry.add("spring.datasource.username", mysql::getUsername);registry.add("spring.datasource.password", mysql::getPassword);// 配置Redisregistry.add("spring.redis.host", redis::getHost);registry.add("spring.redis.port", () -> redis.getMappedPort(6379));}@BeforeEachvoid setup() throws Exception {// 初始化数据库表结构try (Connection conn = dataSource.getConnection();Statement stmt = conn.createStatement()) {stmt.execute("CREATE TABLE IF NOT EXISTS orders (" +"id VARCHAR(36) PRIMARY KEY, " +"product_id VARCHAR(20), " +"quantity INT, " +"status VARCHAR(20))");stmt.execute("TRUNCATE TABLE orders"); // 清空测试数据}// 清空Redis缓存redisTemplate.getConnectionFactory().getConnection().flushAll();}//--------------------------- 业务接口测试 ---------------------------@Testvoid testCreateOrder_ShouldSaveToDatabaseAndCache() {// 构造请求体String requestBody = """{"productId": "P1001","quantity": 2}""";// 设置请求头HttpHeaders headers = new HttpHeaders();headers.setContentType(MediaType.APPLICATION_JSON);headers.setAccept(Collections.singletonList(MediaType.APPLICATION_JSON));HttpEntity<String> request = new HttpEntity<>(requestBody, headers);// 调用接口ResponseEntity<String> response = restTemplate.postForEntity("/api/orders", request, String.class);// 验证HTTP响应assertEquals(HttpStatus.CREATED, response.getStatusCode());assertTrue(response.getBody().contains("orderId"));assertTrue(response.getBody().contains("P1001"));// 验证数据库写入try (Connection conn = dataSource.getConnection();Statement stmt = conn.createStatement();ResultSet rs = stmt.executeQuery("SELECT * FROM orders")) {assertTrue(rs.next());assertEquals("P1001", rs.getString("product_id"));assertEquals(2, rs.getInt("quantity"));} catch (Exception e) {fail("数据库验证失败: " + e.getMessage());}// 验证Redis缓存String cacheKey = "order:" + extractOrderId(response.getBody()); // 假设响应中有orderIdString cachedData = redisTemplate.opsForValue().get(cacheKey);assertNotNull(cachedData);assertTrue(cachedData.contains("\"status\":\"CREATED\""));}//--------------------------- 工具方法 ---------------------------private String extractOrderId(String jsonResponse) {// 简单实现:从JSON中提取orderId(实际项目建议用JSON Path)return jsonResponse.split("\"orderId\":\"")[1].split("\"")[0];}
}
效果:集成测试稳定性提升70%,联调阶段接口问题减少65%。
3. 端到端测试:黄金链路场景化
目标:模拟用户完整业务流程。
误区:试图覆盖100%用户路径,维护成本超出收益。
最佳实践:
- 核心链路筛选:基于用户行为数据(如埋点分析)确定TOP10高频场景;
- 环境治理:使用Testcontainers构建独立Docker环境,避免数据污染。
三、工具链闭环设计
1. 自动化测试流水线架构
阶段 | 工具链 | 关键任务 |
代码提交 | Git | 触发pre-commit钩子检查 |
接口文档生成 | Swagger | 生成 HTML/PDF 格式的 API 文档 |
静态检查 | SonarQube | 代码规范、复杂度分析 |
单元测试 | JUnit5 + JaCoCo | 核心逻辑验证与覆盖率统计 |
集成测试 | Testcontainers | 真实中间件交互验证 |
契约测试 | Spring Cloud Contract | 接口契约一致性校验 |
性能基准测试 | JMeter + Maven插件 | 关键路径负载测试 |
报告生成 | Allure | 测试结果可视化 |
2、契约测试与文档化的协同
痛点:接口频繁变更,导致联调阶段大量阻塞性问题。
解决方案:
- Swagger+Spring Cloud Contract双驱动:
- 通过Swagger定义API规范,生成在线文档供前端参考
- 使用Spring Cloud Contract生成消费者驱动的契约测试用例
3、质量门禁分层设置
检查阶段 | 度量维度 | 阻断阈值 | 工具支撑 |
代码提交 | 单元测试通过率 | 100% | Git Hooks + Surefire |
流水线构建 | 契约测试覆盖率 | ≥95% | Jenkins + Pact Broker |
预发布环境 | 核心链路成功率 | ≥99.9% | Grafana + Prometheus |
四、团队协作升级:测试左移与知识固化
1.需求阶段介入
测试团队参与用户故事拆分,定义明确的验收条件(AC)
2.代码提交卡点
通过Maven Enforcer插件强制前置检查:
<plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-enforcer-plugin</artifactId> <version>3.1.0</version> <executions> <execution> <id>enforce-test-pass</id> <phase>validate</phase> <goals><goal>enforce</goal></goals> <configuration> <rules> <requireTestSuccess> <message>核心用例未通过,禁止提交!</message> </requireTestSuccess> </rules> </configuration> </execution> </executions>
</plugin>
3.可视化看板
使用Grafana监控测试通过率与构建耗时,暴露瓶颈环节
4.知识库沉淀
- 测试模式库:分类归档典型场景(如幂等性验证方案);
- 故障案例库:记录历史缺陷根因与修复方案(支持语义搜索)
五、总结:构建质量提升的“飞轮效应”
通过分层测试策略精准定位漏洞、工具链闭环实现快速反馈、团队协作固化最佳实践,最终形成:缺陷根因分析 → 策略优化 → 工具落地 → 数据验证 → 知识沉淀的质量飞轮。
结语:测试是架构可持续性的基石
测试不仅是保障功能正确性的手段,更是驱动架构演进的重要反馈机制。作为开发人员,只有将测试思维融入编码习惯,才能构建出真正高可用、易扩展的系统。
相关文章:
开发者的测试复盘:架构分层测试策略与工具链闭环设计实战
摘要 针对测试复盘流于形式、覆盖率虚高等行业痛点,本文提出一套结合架构分层与工具链闭环的解决方案: 分层测试策略精准化:通过单元测试精准狙击核心逻辑、契约测试驱动接口稳定性、黄金链路固化端到端场景,实现缺陷拦截率…...
手写CString类
学习和理解字符串处理机制:手写 CString 类是深入学习字符串处理和内存管理的有效方式。通过实现构造函数、析构函数、赋值运算符等,能够理解字符串在内存中的存储方式、动态内存分配和释放的原理,以及如何处理字符串的复制、拼接、查找等操作…...
electron结合vue,直接访问静态文件如何跳转访问路径
在最外的app.vue或者index.vue的js模块编写 let refdade ref(1);//刷新,获得请求// 获取完整的查询字符串(例如: "?dade/myms")const searchParams new URLSearchParams(window.location.search);// 获取 dade 参数的值…...
解读RTOS 第七篇 · 驱动框架与中间件集成
1. 引言 在面向生产环境的 RTOS 系统中,硬件驱动框架与中间件层是连接底层外设与上层应用的桥梁。一个模块化、可扩展的驱动框架能够简化外设管理,提升代码可维护性;而丰富的中间件生态则为网络通信、文件系统、图形界面、安全加密等功能提供开箱即用的支持。本章将从驱动模…...
Java GUI开发全攻略:Swing、JavaFX与AWT
Swing 界面开发 Swing 是 Java 中用于创建图形用户界面(GUI)的库。它提供了丰富的组件,如按钮、文本框、标签等。 import javax.swing.*; import java.awt.event.ActionEvent; import java.awt.event.ActionListener;public class SwingExa…...
Cursor 0.5版本发布,新功能介绍
Cursor,这款流行的AI编程平台,刚刚在其v0.50更新中推出了一系列新功能。 首先,请将您的Cursor IDE更新到最新版本。当您打开Cursor时,您应该会在屏幕左下方收到关于最新版本发布的通知。 更多上下文控制: 对上下文的精细可见性,以及最新模型的MAX模式。 聊天升级: 导出…...
Android学习总结之Glide自定义三级缓存(实战篇)
一、为什么需要三级缓存 内存缓存(Memory Cache) 内存缓存旨在快速显示刚浏览过的图片,例如在滑动列表时来回切换的图片。在 Glide 中,内存缓存使用 LruCache 算法(最近最少使用),能自动清理长…...
Maven 下载安装与配置教程
## 1. Maven 简介 Maven 是一个项目管理和构建自动化工具,主要用于 Java 项目。Maven 可以帮助开发者管理项目的构建、报告和文档,简化项目依赖管理。 ## 2. 下载 Maven 1. 访问 Maven 官方网站 [https://maven.apache.org/download.cgi](https://maven.…...
一篇解决Redis:持久化机制
目录 认识持久化 持久化方案 RDB(Redis DataBase) 手动触发 自动触发 小结 AOF(Append-Only File) AOF缓冲区刷新机制 AOF重写机制 AOF重写流程 编辑 混合持久化 认识持久化 我们都知道Mysql有四大特征,原子性,持久…...
使用IDEA创建Maven版本的web项目以及lombok的使用
1.新建项目 2.修改pom.xml 3.修改项目结构 4.在main/java下面写一个Servlet测试一下 然后当前页面往下滑 -Dfile.encodingUTF-8编写一句输出语句,测试是否成功部署配置,并选择到正确的位置: 回车以后 再回到idea里面,发现控…...
2025年AI开发者在开发者占比?
AI开发者在全球开发者中的占比目前没有一个统一且精确的数值,但根据行业报告和调研数据,可以给出以下大致的范围和趋势分析: 1. 综合估算范围 全球范围:AI/ML(机器学习)开发者约占开发者总数的 5%-15%&…...
SpringBoot整合MQTT实战:基于EMQX构建高可靠物联网通信,从零到一实现设备云端双向对话
一、引言 随着物联网(IoT)技术的快速发展,MQTT(Message Queuing Telemetry Transport)协议因其轻量级、低功耗和高效的特点,已成为物联网设备通信的事实标准。本文将详细介绍如何使用SpringBoot框架整合MQTT协议,基于开源MQTT代理EMQX实现设…...
Windows更新暂停七天关键注册表
环境:windows10 工具:procmon 下载地址:https://learn.microsoft.com/zh-cn/sysinternals/downloads/procmon 监控截图: 界面截图: 注: 1.北京时间差8小时 2.至少是从第二天恢复,即至少暂停…...
小白学习java第18天(上):spring
Spring :是一个轻量级(一个小依赖就可以实现还不是轻量级)的控制反转(IOC)和面向切面编程(AOP)的框架! 优点: 1.Spring 是一个开源免费的框架(容器…...
用Array.from实现创建一个1-100的数组
一、代码实现 let arr Array.from({length: 100}, (_, i) > i 1); 二、代码分析 1、Array.from(arrayLike, mapFn) (1)arrayLike 类数组对象(如 { length: 100 })本身没有索引属性(如 0: undefined, 1: undefi…...
什么是物联网 IoT 平台?
目录 物联网IoT平台的定义 物联网 IoT 平台发展历程 物联网IoT平台数据的特征 物联网IoT平台的处理流程 专为物联网 IoT 平台处理而生的时序数据库 物联网 IoT 平台时序数据处理面临的挑战及解决方案 收益与价值 物联网 IoT 平台企业案例 至数摇光 x TDengine 华自科技…...
PostgREST:无需后端 快速构建RESTful API服务
在现代 Web 开发中,API 已成为连接前后端的核心桥梁,传统的做法是通过后端框架来构建API接口,然后由前后端人员进行联调。 PostgREST是基于无服务器的一种实现方案,允许开发者将PostgreSQL数据库直接暴露为RESTful API࿰…...
3天北京旅游规划
北京 第一天应该集中在故宫和市中心区域,比如天安门、人民广场。这样可以体验到北京的历史和政治文化。午餐推荐烤鸭,因为这可是北京的特色。下午可以安排南锣鼓巷,既有古色古香的胡同,又有丰富的美食选择。 第二天的话࿰…...
MySQL--day1--数据库概述
(以下内容全部来自上述课程) 概述 1. 为什么要用数据库 持久化:内存中的数据断电之后就不存在了,所以需要持久化–>需要相关介质。 其中的一个介质就是数据库:存储数据量大、存储数据类型多 2. 数据库与数据库…...
[思维模式-38]:看透事物的关系:什么是事物的关系?事物之间的关系的种类?什么是因果关系?如何通过数学的方式表达因果关系?
一、什么是事物的关系? 事物的关系是指不同事物之间存在的各种联系和相互作用,它反映了事物之间的相互依存、相互影响、相互制约等特性。以下从不同维度为你详细阐述: 1、关系的类型 因果关系 定义:一个事件(原因&a…...
图像识别与 OCR 应用实践
图像识别是一种让计算机具备“看”与“理解”图像能力的人工智能技术,其目标是从图像或视频中提取有意义的信息,如物体、人物、场景或文字。在现实生活中,这项技术被广泛应用于面部识别、自动驾驶、安防监控、医疗诊断、图像搜索等多个领域。…...
深入理解卷积神经网络:从基础原理到实战应用
在人工智能领域,卷积神经网络(Convolutional Neural Network,简称 CNN)凭借其强大的图像识别、处理能力,成为深度学习中不可或缺的技术。无论是自动驾驶汽车识别道路标志,还是医学影像分析辅助疾病诊断&…...
51单片机——交通指示灯控制器设计
设计目标 1、设计一交通灯控制,控制东西方向的红、黄、绿灯和南北方向的红、黄、绿灯。 2、可手动控制和自动控制,设置两个输入控制开关。 手动/自动开关,通过P11的按键输入控制 3、手动:设置开关P11,两种情况&#x…...
vue2 头像上传+裁剪组件封装
背景:最近在进行公司业务开发时,遇到了头像上传限制尺寸的需求,即限制为一寸证件照(宽295像素,高413像素)。 用到的第三方库: "vue-cropper": "^0.5.5" 完整组件代码&…...
面向对象设计模式之代理模式详解
文章目录 面向对象设计模式之代理模式详解面向对象思想:现代软件开发的基石代理模式:巧妙的中间层设计JavaScript 语法点与代理模式的结合JavaScript 实现代理模式示例代理模式的应用场景 面向对象设计模式之代理模式详解 在现代软件开发的浩瀚领域中&a…...
Leetcode209做题笔记
力扣209 题目分析:想象一个窗口遍历着这个数组,不断扩大右边界,让r。往窗口中添加数字: 此时我们找到了这个窗口,它的和满足了大于等于target的条件,题目让我求最短的,那么我们就尝试来缩短它&…...
SVG 知识详解:从入门到精通
SVG 知识详解:从入门到精通 作为一名前端开发者,我经常会被SVG的魅力所折服。这种基于XML的矢量图形格式,不仅能完美适配各种屏幕分辨率,还能通过CSS和JavaScript进行灵活控制。今天,就让我们一起来深入探索SVG的世界…...
编译openssl源码
openssl版本 1.1.1c windows 安装环境 perl 先安装perl,生成makefile需要 https://strawberryperl.com/releases.html nasm nasm 也是生成makefile需要 https://www.nasm.us/ 安装完perl输入一下nasm,看看能不能找到,找不到的话需要配…...
土壤温湿盐分传感器用于节水农业灌溉引领者三针设计原理便于安装维护
土壤温度部分是由精密铂电阻和高精度变送器两部分组成。变送器部分由电源模块、温度传感模块、变送模块、温度补偿模块及数据处理模块等组成,彻底解决铂电阻因自身特点导入的测量误差,变送器内有零漂电路和温度补偿电路,对使用环境有较高的适…...
Kotlin Compose 与传统 Android UI 开发对比
在移动应用开发领域,Android 开发一直是技术演进的前沿阵地,而 UI 开发作为用户与应用交互的核心环节,其技术体系的变革更是备受瞩目。 技术演进背景 Android UI 开发体系发展脉络 原生 View 体系阶段 在早期的 Android 开发中,原生 View 体系占据了主导地位。开发者通…...
docker-compose——安装redis
文章目录 一、编写docker-compose.yaml文件二、编写redis.conf文件三、启动docker-compose 一、编写docker-compose.yaml文件 version: 3.3 services:redis:image: redis:latestcontainer_name: redisrestart: alwaysports:- 6379:6379volumes:- ./redis/data:/data- ./redis/…...
MFC 调用海康相机进行软触发
初始化相机类文件 #pragma once #include "MvCameraControl.h" class CMvCamera { public:CMvCamera();~CMvCamera();//初始化相机int InitCamera();int SaveCurrentImage(CString filePath);//关闭相机void CloseCamera();//设置int SetEnumValue(IN const char* s…...
第二章 变量和运算符
主要内容 关键字和标识符变量和常量八大基本数据类型Scanner键盘输入基本数据类型的类型转换算术运算符赋值运算符扩展赋值运算符比较运算符逻辑运算符三目运算符运算符的优先级别 学习目标 知识点要求关键字和标识符理解变量和常量掌握八大基本数据类型掌握Scanner键盘输入…...
【starrocks】StarRocks 常见 HTTP 操作与导入错误排查指南
文章目录 一、Stream Load:通过 HTTP 导入数据二、导入状态查询三、取消导入任务四、节点状态监控查看所有 Backend 状态:查看所有 Frontend 状态: 五、导入失败的排查方式1. 查询导入任务状态2. 下载详细错误日志3. 查看 FE/BE 节点日志FE 日…...
网络协议分析 实验七 FTP、HTTP、DHCP
文章目录 实验7.1 FTP协议练习二 使用浏览器登入FTP练习三 在窗口模式下,上传/下传数据文件实验7.2 HTTP(Hyper Text Transfer Protocol)练习二 页面提交练习三 访问比较复杂的主页实验7.3 DHCP(Dynamic Host Configuration Protocol) 实验7.1 FTP协议 dir LIST&…...
ET ProcessInnerSender类(实体) 分析
ProcessInnerSender 作用是进程内部发送Actor消息 字段 TIMEOUT_TIME 超时时间RpcId 用来累加requestCallback 存储RPC的回调事件list 用来获取MessageQueue中的Actor消息 方法 Awake 初始化在MessageQueue中注册待处理的消息队列Destroy 移除在MessageQueue中的消息队列U…...
远程连接工具
绿色轻便ToDesk https://www.todesk.com/download.html 向日葵 https://sunlogin.oray.com/download...
MySQL库级管理:数据库管理与存储引擎剖析
引言 各位数据库爱好者们好!今天我们要深入探讨MySQL数据库的基本操作,这是每位开发者必须掌握的"内功心法" 💪。无论你是刚接触MySQL的小白,还是需要复习基础的老手,这篇教程都将带你系统学习数据库的核心…...
MongoDB 的核心概念(文档、集合、数据库、BSON)是什么?
MongoDB 是一个面向文档的数据库,它的核心概念与传统的关系型数据库(RDBMS)有所不同。以下是它的四个主要核心概念: 文档 (Document) 定义: 文档是 MongoDB 中的基本数据单元。它类似于关系型数据库中的一行记录&#…...
占位符读取标准输入缓冲区规则
1、如果标准输入缓冲区中的前若干个字符都是空白字符,%s,%d,%f都能直接跳过并且从第一个非空白字符开始读取,但%c不能,而是直接读取。 2、%s遇到空白字符时停止,不会读取遇到的空白字符。 3、%d遇到非数字…...
使用Docker部署MongoDB
使用Docker部署MongoDB 使用Docker部署MongoDB是一种快速、便捷的方式,以下是详细步骤: 1. 基本部署 拉取MongoDB镜像 docker pull mongo:latest运行MongoDB容器(最简单方式) docker run --name mongodb -d -p 27017:27017 m…...
如何通过命令提示符(CMD)检查虚拟化是否已启用
想确认 Windows 11 或 10 电脑或笔记本上的虚拟化是否已启用,可以使用命令提示符或 PowerShell 中的 systeminfo 命令。之前已经介绍过使用图形界面的检查方法。在大多数现代系统中,虚拟化默认在 BIOS 中启用,但我们无需打开 BIOS/UEFI 固件菜单即可确认这一点。 虚拟化是一…...
【氮化镓】AlGaN合金中成分相关的辐射响应
2025年,美国宾夕法尼亚州立大学Miaomiao Jin等人基于分子动力学模拟方法,研究了AlGaN合金中成分依赖的辐射响应。研究结果表明,AlGaN合金的辐射损伤特性与铝含量密切相关:随铝含量增加,单次碰撞事件产生的缺陷减少,但由于累积效应,富铝系统更易形成扩展间隙缺陷簇,导致…...
『 测试 』软件测试全流程与Bug管理核心要点解析
文章目录 1 软件测试生命周期2 Bug2.1 Bug 的概念2.2 提出清晰的 Bug2.3 Bug 级别2.4 Bug 的生命周期2.5 作为测试人员与开发人员发生冲突怎么办 1 软件测试生命周期 软件测试贯穿于软件的整个生命周期; 需求分析 测试前需要对需求进行分析, 分析通常站在三个角度去考虑, 即用…...
【springcloud学习(dalston.sr1)】使用Feign实现接口调用(八)
该系列项目整体介绍及源代码请参照前面写的一篇文章【springcloud学习(dalston.sr1)】项目整体介绍(含源代码)(一) (一)Feign的理解 前面文章【springcloud学习(dalston.sr1)】服务消费者通过restTemplat…...
嵌入式Linux Qt开发:2、Qt creator简单配置、Qt Designer使用以及信号槽机制使用
一、前言 Qt creator比较轻量,而且是Qt自带的IDE,基本可以开箱即用,个人使用起来感觉还是比较舒服的,并且其自带的FakeVim也可以做一些Vim的配置,其主界面可以简单配置显示很多的窗口,让开发更加顺畅。 Q…...
【Vue 3 + Vue Router 4】如何正确重置路由实例(resetRouter)——避免“VueRouter is not defined”错误
在使用 Vue 3 和 Vue Router 4 开发中大型 SPA 应用时,我们经常会遇到需要动态添加或删除路由的场景。尤其是在权限控制和用户登出后重置路由的需求中,正确地实现 resetRouter 非常关键。 然而,许多开发者在迁移或初始化项目时,会…...
企业内部通讯,企业级即时通讯软件选择
企业内部的沟通往往涉及大量敏感信息,如商业机密、财务数据、客户信息等。BeeWorks IM即时通讯平台采用全私有化部署,企业可以将服务器部署在自己的数据中心或私有云环境中,确保所有数据的存储和传输都在企业可控的网络内完成。这种部署方式从…...
300. 最长递增子序列
理解最长递增子序列(LIS)是解决该问题的关键。子序列是从给定数组中按顺序选取的元素序列,例如数组 [1, 2, 3, 4, 5] 的子序列可以是 [2, 3, 4]。需要注意的是,子序列的元素在原数组中不一定是连续的。因此,最长递增子…...
MongoDB入门
1.MongoDB 基本概念详解 2.MongoDB 快速实战 3.MongoDB 核心操作与原理详解 Mongo 是 humongous 的中间部分,在英文里是“巨大无比”的意思。所以 MongoDB 可以翻译 成“巨大无比的数据库”,更优雅的叫法是“海量数据库”。Mongodb是一款非关系型数据库…...