架构思维:构建高并发读服务_基于流量回放实现读服务的自动化测试回归方案

文章目录

  • 引言
  • 一、升级读服务架构,为什么需要自动化测试?
  • 二、自动化回归测试系统:整体架构概览
  • 三、日志收集
    • 1. 拦截方式
    • 2. 存储与优化策略
    • 3. 架构进化
  • 四、数据回放
    • 技术实现
    • 关键能力
  • 五、差异对比
    • 对比方式
    • 灵活配置
  • 六、三种回放模式详解
    • 1. 离线回放
    • 2. 实时回放(对比新旧服务)
    • 3. 无录制实时回放
  • 七、使用注意事项与最佳实践
  • 八、模拟核心Code
  • 九、小结

在这里插入图片描述

引言

在高并发读服务的架构优化过程中,我们往往关注系统如何抗压、如何缓存命中率更高,甚至在性能提升方案落实后迅速投入重构。然而,在这一过程中,容易被忽略的一环就是“测试回归”。

接下来我们将从实际落地角度,系统性地介绍一种支持读服务快速升级、业务稳定推进的「自动化测试回归系统架构」方案,构建一套覆盖全量场景、支持自助回归的自动化测试体系。


一、升级读服务架构,为什么需要自动化测试?

假设我们已落地了支持高并发的读服务架构,包括懒加载缓存、全量缓存、数据同步机制等。但读服务的升级改造带来的“回归压力”却是另一种挑战:

  • 架构重构往往影响范围广,测试周期按“月”计。
  • 日常需求中即便仅修改部分接口逻辑,也可能因底层复用代码影响其他未修改接口,造成线上 Bug。

新老版本的接口未变架构图

在这里插入图片描述

解决这类回归测试痛点,最优解是:自动化测试系统。这不仅提升了测试效率,也为系统升级提供了「安全缓冲带」。


二、自动化回归测试系统:整体架构概览

自动化测试回归系统的核心由三个模块组成:

  1. 日志收集:拦截线上请求,记录请求参数和响应数据,生成“真实用户用例”。
  2. 数据回放:基于收集的请求数据,自动向新旧服务发起请求,触发真实的业务流程。
  3. 差异对比:将新老版本的响应结果进行对比,捕捉潜在 Bug。

在这里插入图片描述


三、日志收集

基于过滤器的日志收集架构图

在这里插入图片描述

1. 拦截方式

  • HTTP 接口:基于 Spring 的 Interceptor 或 Servlet 的 Filter 拦截。
  • RPC 接口:拦截 RPC 框架底层通信逻辑(如 Dubbo 的 Filter)。

拦截的请求被封装为统一格式:

{"应用名": "XXX","接口方法名": "/api/order/detail","入参": "{...}","出参": "{...}"
}

这些日志通过 MQ 推送至回归平台进行存储与处理。

2. 存储与优化策略

  • 接口元数据存储在关系型数据库
  • 大体量出入参数据存储在如 HBase 等高吞吐的 NoSQL
  • 提供去重、清洗、采样功能,避免数据爆炸性增长
  • 非业务环境如压测数据需剔除

3. 架构进化

单独进程的日志收集架构图

在这里插入图片描述

  • 同进程采集:轻量集成,但存在侵入性
  • 独立进程采集:将日志打印至文件,由单独进程监听并推送 MQ,降低业务系统资源占用

四、数据回放

数据回放模块模拟用户请求,通过原始日志数据中的入参信息,重放请求以获得当前版本的响应数据。

在这里插入图片描述

技术实现

  • HTTP 接口:使用 RestTemplate、OkHttp、Apache HttpClient 等发起请求
  • RPC 接口:基于 RPC 框架提供的泛化调用能力构造请求

关键能力

  • 多线程并发执行,支持批量回放
  • 回放任务可手动触发,也可通过策略定时运行
  • 支持失败重试与限流策略,避免压垮被测服务

五、差异对比

对比模块将原始接口返回值与当前版本的回放结果进行对比,判断是否存在行为变更。

对比方式

  • 文本对比(推荐):将返回结果转为 JSON 结构,逐字段比对
  • 校验和对比(不推荐):判断整体一致性但缺乏定位能力

灵活配置

  • 支持忽略字段配置(如 UUID、traceId)
  • 支持字段级别容差设置(如时间戳误差容忍 3s)

六、三种回放模式详解

不同业务阶段与环境下,可以灵活选择回放模式:

1. 离线回放

仅调用新版本服务,使用历史日志作为“期望值”。

在这里插入图片描述

优点:不影响线上系统
缺点:若数据发生变化,对比结果失效


2. 实时回放(对比新旧服务)

手动触发后,分别调用新旧版本,实时比较返回数据。

在这里插入图片描述

优点:规避数据变更问题,结果更真实
缺点:两次调用增加系统负载


3. 无录制实时回放

完全实时处理:日志一进入消费队列即触发双版本回放与对比。

在这里插入图片描述

优点:最强覆盖率,避免重要场景被采样遗漏
缺点:性能压力较大,需在资源允许下谨慎使用


七、使用注意事项与最佳实践

  • 屏蔽写接口:避免写接口等副作用操作回放引发业务混乱
  • 合理限流:对线上环境回放要设限流阈值
  • 数据存储生命周期:入参/出参数据定期清理,避免存储崩溃
  • 差异字段管控:灵活配置忽略项,避免误报
  • 自动告警机制:支持对比失败数据告警与可视化管理

八、模拟核心Code

package com.example.autotest;import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;@SpringBootApplication
public class AutoTestApplication {public static void main(String[] args) {SpringApplication.run(AutoTestApplication.class, args);}
}// 1. 日志收集 - Spring Interceptor
package com.example.autotest.interceptor;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import com.alibaba.fastjson.JSON;
import org.springframework.beans.factory.annotation.Autowired;
import com.example.autotest.mq.LogProducer;@Component
public class LoggingInterceptor implements HandlerInterceptor {@Autowiredprivate LogProducer logProducer;@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 记录请求时间、请求体等RequestContextHolder.start();return true;}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {LogRecord record = new LogRecord();record.setAppName("order-service");record.setApi(request.getRequestURI());record.setRequestBody(RequestContextHolder.getRequestBody());record.setResponseBody(RequestContextHolder.getResponseBody());// 推送到 MQlogProducer.send(JSON.toJSONString(record));}
}// 2. MQ 生产者
package com.example.autotest.mq;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.kafka.core.KafkaTemplate;
import org.springframework.stereotype.Component;@Component
public class LogProducer {@Autowiredprivate KafkaTemplate<String, String> kafkaTemplate;public void send(String message) {kafkaTemplate.send("auto-test-logs", message);}
}// 3. 日志消费 & 回放触发
package com.example.autotest.consumer;import org.springframework.kafka.annotation.KafkaListener;
import org.springframework.stereotype.Service;
import com.alibaba.fastjson.JSON;
import com.example.autotest.model.LogRecord;
import com.example.autotest.replay.ReplayService;@Service
public class LogConsumer {private final ReplayService replayService;public LogConsumer(ReplayService replayService) {this.replayService = replayService;}@KafkaListener(topics = "auto-test-logs")public void listen(String message) {LogRecord record = JSON.parseObject(message, LogRecord.class);// 异步触发回放replayService.submit(record);}
}// 4. 回放服务
package com.example.autotest.replay;import com.example.autotest.model.LogRecord;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;
import com.alibaba.fastjson.JSON;@Service
public class ReplayService {private final ExecutorService executor = Executors.newFixedThreadPool(20);private final RestTemplate restTemplate = new RestTemplate();public void submit(LogRecord record) {executor.submit(() -> {// 调用旧版接口String oldRes = restTemplate.postForObject(record.getApi(), record.getRequestBody(), String.class);// 调用新版接口(假设前缀不同)String newApi = record.getApi().replace("/v1/", "/v2/");String newRes = restTemplate.postForObject(newApi, record.getRequestBody(), String.class);// 对比结果DiffResult diff = JsonDiffComparator.compare(record.getResponseBody(), newRes);if (!diff.isEqual()) {// 记录差异或报警System.err.println("Data mismatch on API " + record.getApi() + ": " + diff.getDetails());}});}
}// 5. 差异对比工具
package com.example.autotest.replay;import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.flipkart.zjsonpatch.JsonDiff;public class JsonDiffComparator {private static final ObjectMapper mapper = new ObjectMapper();public static DiffResult compare(String expectedJson, String actualJson) {try {JsonNode e = mapper.readTree(expectedJson);JsonNode a = mapper.readTree(actualJson);JsonNode patch = JsonDiff.asJson(e, a);if (patch.size() == 0) {return new DiffResult(true, null);}return new DiffResult(false, patch.toString());} catch (Exception ex) {throw new RuntimeException(ex);}}
}// 6. 差异结果模型
package com.example.autotest.replay;public class DiffResult {private final boolean equal;private final String details;public DiffResult(boolean equal, String details) {this.equal = equal;this.details = details;}public boolean isEqual() { return equal; }public String getDetails() { return details; }
}

九、小结

通过日志收集、数据回放和差异对比三大模块的组合,读服务的测试回归过程实现了自动化、精细化、可视化,彻底摆脱“人工全量测试回归”的低效流程,极大地提升了系统重构与业务迭代的安全性与效率。

在这里插入图片描述

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

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

相关文章

基于Spring Boot 3.0、ShardingSphere、PostgreSQL或达梦数据库的分库分表

要实现基于Spring Boot 3.0、ShardingSphere、PostgreSQL或达梦数据库的分库分表&#xff0c;首先需要对ShardingSphere进行一些基本配置。你提到的溯源码、批次号等数据需要考虑到跨年数据的存储&#xff0c;因此要设计一个能够动态扩展的分表策略 添加ShardingSphere依赖 在…

位运算的应用

1. 判断偶数&#xff0c;判断最低位是0还是1即可&#xff0c;⽐求模快 x % 2 ! 0 //x正负都可以判断&#xff1b;不⽤x%2 1&#xff0c;因为如果x为负奇数&#xff0c;x%2-1 (x & 0x1) 0 例如&#xff1a; int x; int main() { cin>>x; if((x & 0x1)0) cout<…

FOC算法开环控制基础

1. 为什么要有FOC算法 先看看从有刷电机到无刷电机的简单介绍&#xff0c;如下图1&#xff0c;通电螺线圈会产生磁场&#xff0c;这个磁场会产生N级和S级&#xff0c;然后这个电磁铁就可以吸引永磁体&#xff0c;S级吸引N级&#xff0c;N级吸引S级&#xff0c;通俗的来说&…

【计算机网络】HTTP中GET和POST的区别是什么?

从以下几个方面去说明&#xff1a; 1.定义 2.参数传递方式 3.安全性 4.幂等性 1.定义&#xff1a; GET&#xff1a; 获取资源&#xff0c;通常请求数据而不改变服务器的状态。POST&#xff1a; 提交数据到服务器&#xff0c;通常会改变服务器的状态或副作用(如创建或更新资源…

7400MB/s5050TBW完美结合,全新希捷酷玩530R SSD体验评测

7400MB/s&5050TBW完美结合&#xff0c;全新希捷酷玩530R SSD体验评测 哈喽小伙伴们好&#xff0c;我是Stark-C~ 说到希捷酷玩530 SSD&#xff0c;很多硬核进阶玩家应该都知道&#xff0c;或者说正在使用&#xff08;比如说我~&#xff09;。 作为希捷大厂旗下高性能SSD的…

(undone) MIT6.S081 2023 学习笔记 (Day11: LAB10 mmap)

url: https://pdos.csail.mit.edu/6.1810/2023/labs/mmap.html mmap和munmap系统调用允许UNIX程序对其地址空间进行精细控制。它们可用于进程间共享内存、将文件映射到进程地址空间&#xff0c;并作为用户级页面错误处理方案的一部分&#xff0c;例如课程中讨论的垃圾回收算法。…

Q_OBJECT宏的作用

Qt 中&#xff0c;如果一个类中定义了信号&#xff08;signals&#xff09;或槽&#xff08;slots&#xff09;&#xff0c;那么这个类必须包含 Q_OBJECT 宏。 Q_OBJECT宏是 Qt 元对象系统的核心部分&#xff0c;它使得信号和槽机制能够正常工作。 Q_OBJECT宏是 Qt 的元对象系统…

信息安全基石:加解密技术的原理、应用与未来

信息加解密技术是信息安全领域的核心技术之一&#xff0c;以下为你详细介绍&#xff1a; 一、加密技术 1.定义&#xff1a;加密是通过特定的算法和密钥&#xff0c;将原始的明文信息转化为看似无意义的密文信息的过程。这一过程使得信息在传输、存储等过程中&#xff0c;即使…

LeetCode:返回倒数第k个结点

1、题目描述 实现一种算法&#xff0c;找出单向链表中倒数第 k 个节点。返回该节点的值。 注意&#xff1a;本题相对原题稍作改动 示例&#xff1a; 输入&#xff1a; 1->2->3->4->5 和 k 2 输出&#xff1a; 4 说明&#xff1a; 给定的 k 保证是有效的。 2、…

R004 -计算机硬件基础

目录 1.数据表示&计算机网络组成 2.计算机网络分类 3.冯诺依曼体系结构 4.指令系统基础 5.指令系统类型 6.流水线技术 流水线周期 &#xff1a;各流水段中&#xff0c;执行时间最长的那一段。就是T 流水线时间&#xff1a;t 1t2t 3 (n-1) * T 7.流水线指标 8.存储系…

Mybatis学习(下)

目录 1. 动态sql的应用 1.2 1.2 1.3 、 、 标签 1.4 1. 动态sql的应用 使用Mybatis框架时, 对于sql数据的操作量比较大的时候, 看着会觉得很乱, 可能写着写着就乱了, 或者说回过头来发现sql语句写错了, 很麻烦, 所以动态sql就可以让我们用Java代码, 替换部分sql语句 1.2 &l…

iview 老版本合并单元格

新版的iview中已经支持了合并单元格了&#xff0c;我的版本比较老&#xff0c;为&#xff1a;"iview": "^3.5.2"。暂不支持。记录一下别的大佬的方法。感觉思路比较活&#xff0c;正在这种思路需要在解决问题的过程中学习。 核心思路&#xff1a;通过rende…

FGMRES(Flexible Generalized Minimal Residual)方法

FGMRES&#xff08;Flexible Generalized Minimal Residual&#xff09;方法是GMRES的变种&#xff0c;主要用于处理变预处理子&#xff08;即每次迭代的预处理子可能不同&#xff09;的情况。与标准GMRES相比&#xff0c;FGMRES通过存储预处理后的向量而非预处理子本身&#x…

自主采集高质量三维重建数据集指南:面向3DGS与NeRF的图像与视频拍摄技巧【2025最新版!!】

一、✨ 引言 随着三维重建技术的飞速发展&#xff0c;NeRF&#xff08;Neural Radiance Fields&#xff09;与 3D Gaussian Splatting&#xff08;3DGS&#xff09;等方法成为重建真实场景和物体几何细节的前沿方案。这些方法在大规模场景建模、机器人感知、文物数字化、工业检…

HarmonyOS Next-DevEco Studio(5.0.2)无网络环境配置(详细教程)

开发者如果电脑处于完全无网环境&#xff0c;可以参考下面文档进行相关配置 DevEco Studio(5.0.2)开发环境一览&#xff1a; 工具版本DevEco Studio5.0.2openHarmonySDK14ohpm5.0.11node.js18.20.1hypium1.0.21 一、下载DevEco Studio&#xff08;5.0.2 Release&#xff09;…

MIT XV6 - 1.1 Lab: Xv6 and Unix utilities - sleep 是怎样练成的?

接上文MIT XV6 - 1.1 Lab: Xv6 and Unix utilities - sleep 探究sleep.c是如何’炼成’的? 老实讲&#xff0c;我不熟悉Makefile&#xff0c;最多写过简单的编译和辅助脚本&#xff0c;拿到Xv6的Makefile是一脸懵的&#xff0c;至今还是一脸懵&#xff0c;那么我们上篇中新加的…

顺序结构双链表的实现

双链表是用最快的时间实现链表的一种方式&#xff0c;具体的实现代码如下&#xff1a; #pragma once #include<stdio.h> #include<stdlib.h> #include<assert.h>typedef int LTDataType; typedef struct ListNode {LTDataType data;struct ListNode* next;/…

GoFrame 奉孝学习笔记

第一章节 GoFrame 是一款基础设施建设比较完善的模块化框架 GoFrame 是一款基础设施建设比较完善的模块化框架, Web Server 模块是其中比较核心的模块,我们这里将 Web 服务开发作为框架入门的选择,便于大家更容易学习和理解。 用GOland编写代码 go.mod module goframePro…

pinia实现数据持久化插件pinia-plugin-persist-uni

在学习uniapp过程中&#xff0c;看到了pinia-plugin-persist-uni插件&#xff0c;以前面试过程中也有面试过说vuex数据刷新之前的数据就丢失了&#xff0c;之前回答的是把数据存储到数据库或者本地存储。pinia-plugin-persist-uni本质上数据也是本地存储。 1、安装 npm instal…

Git 多账号切换及全局用户名设置不生效问,GIT进行上传无权限问题

解决 Git 多账号切换及全局用户名设置不生效问题 在软件开发过程中&#xff0c;我们经常会使用 Git 进行版本控制。有时&#xff0c;我们需要在同一台机器上管理多个 Git 账号&#xff0c;最近我在进行使用git的时候因为项目要进行上传的不同的git账号&#xff0c;但是通过本地…