数据留痕的方法
在项目中,数据变更时,经常需要记录上次的数据,以便查看对比,专业术语叫做数据留痕。数据变更留痕(即记录数据的变更历史)是一个常见的需求,例如在审计、追踪数据变化或满足合规性要求的场景中。以下是数据留痕几种常见的实现方式:
1. 手动记录变更日志
在业务代码中手动记录数据变更的日志,将变更前后的数据保存到日志表或日志文件中。
实现步骤:
-
在数据变更的地方(如更新、删除操作)手动记录变更前后的数据。
-
将变更信息保存到数据库的日志表或日志文件中。
示例代码:
public class UserService {@Autowiredprivate UserRepository userRepository;@Autowiredprivate AuditLogRepository auditLogRepository;public void updateUser(User newUser) {// 获取旧数据User oldUser = userRepository.findById(newUser.getId()).orElseThrow();// 更新数据userRepository.save(newUser);// 记录变更日志AuditLog auditLog = new AuditLog();auditLog.setAction("UPDATE");auditLog.setEntityName("User");auditLog.setEntityId(newUser.getId());auditLog.setOldValue(oldUser.toString()); // 旧数据auditLog.setNewValue(newUser.toString()); // 新数据auditLog.setChangeTime(new Date());auditLogRepository.save(auditLog);}
}
优点:
-
实现简单,直接控制日志内容。
-
灵活性高,可以根据需求定制日志格式。
缺点:
-
代码侵入性强,需要在每个变更点手动添加日志记录。
-
容易遗漏,维护成本较高。
2. 使用AOP(面向切面编程)
通过AOP在数据变更的方法上添加切面,自动记录变更日志。
实现步骤:
-
定义一个切面,拦截数据变更的方法(如
update、delete)。 -
在切面中获取方法的参数和返回值,记录变更前后的数据。
示例代码:
@Aspect
@Component
public class DataChangeAspect {@Autowiredprivate AuditLogRepository auditLogRepository;@AfterReturning(pointcut = "execution(* com.example.service.UserService.updateUser(..))", returning = "result")public void logDataChange(JoinPoint joinPoint, Object result) {Object[] args = joinPoint.getArgs();User newUser = (User) args[0]; // 获取新数据User oldUser = (User) result; // 获取旧数据// 记录变更日志AuditLog auditLog = new AuditLog();auditLog.setAction("UPDATE");auditLog.setEntityName("User");auditLog.setEntityId(newUser.getId());auditLog.setOldValue(oldUser.toString()); // 旧数据auditLog.setNewValue(newUser.toString()); // 新数据auditLog.setChangeTime(new Date());auditLogRepository.save(auditLog);}
}
优点:
-
代码侵入性低,集中管理日志逻辑。
-
灵活,可以根据需求定制切面。
缺点:
-
需要熟悉AOP编程。
-
可能增加系统复杂性。
3. 使用数据库触发器
通过数据库触发器在数据变更时自动记录历史数据。
实现步骤:
-
在数据库中创建触发器,监听目标表的变更(如
INSERT、UPDATE、DELETE)。 -
在触发器中将变更前后的数据插入到历史表中。
示例 SQL:
CREATE TABLE users_history (id INT PRIMARY KEY AUTO_INCREMENT,user_id INT,name VARCHAR(255),email VARCHAR(255),action VARCHAR(10),change_time TIMESTAMP
);CREATE TRIGGER trg_user_history
AFTER UPDATE ON users
FOR EACH ROW
BEGININSERT INTO users_history (user_id, name, email, action, change_time)VALUES (OLD.id, OLD.name, OLD.email, 'UPDATE', NOW());
END;
优点:
-
与应用程序解耦,数据库层面实现。
-
无需修改业务代码。
缺点:
-
触发器可能影响数据库性能。
-
调试和维护复杂。
4. 使用事件监听机制
通过 Spring 的事件监听机制,在数据变更时发布事件并记录日志。
实现步骤:
-
定义一个事件类(如
DataChangeEvent)。 -
在数据变更的地方发布事件。
-
监听事件并记录日志。
示例代码:
事件类:
@Data
public class DataChangeEvent {private String entityName;private Long entityId;private String oldValue;private String newValue;}
发布事件:
@Service
public class UserService {@Autowiredprivate ApplicationEventPublisher eventPublisher;public void updateUser(User newUser) {User oldUser = userRepository.findById(newUser.getId()).orElseThrow();userRepository.save(newUser);// 发布事件DataChangeEvent event = new DataChangeEvent("User", newUser.getId(), oldUser.toString(), newUser.toString());eventPublisher.publishEvent(event);}
}
监听事件:
@Component
public class DataChangeListener {@Autowiredprivate AuditLogRepository auditLogRepository;@EventListenerpublic void handleDataChangeEvent(DataChangeEvent event) {AuditLog auditLog = new AuditLog();auditLog.setAction("UPDATE");auditLog.setEntityName(event.getEntityName());auditLog.setEntityId(event.getEntityId());auditLog.setOldValue(event.getOldValue());auditLog.setNewValue(event.getNewValue());auditLog.setChangeTime(new Date());auditLogRepository.save(auditLog);}
}
优点:
-
解耦业务逻辑和日志记录。
-
灵活,支持异步处理。
缺点:
-
需要熟悉 Spring 事件机制。
-
可能增加系统复杂性。
5.总结
| 方式 | 优点 | 缺点 | 适用场景 |
|---|---|---|---|
| 手动记录日志 | 简单直接,灵活性高 | 代码侵入性强,维护成本高 | 小型项目,简单需求 |
| AOP | 代码侵入性低,集中管理日志逻辑 | 需要熟悉 AOP,可能增加复杂性 | 需要集中管理日志的中大型项目 |
| 数据库触发器 | 与应用程序解耦,无需修改代码 | 调试复杂,可能影响性能 | 数据库层面的审计需求 |
| 事件监听机制 | 解耦业务逻辑,支持异步处理 | 需要熟悉 Spring 事件机制 | 需要解耦和异步处理的场景 |
根据项目需求和技术栈选择合适的方式。
如果项目使用 Hibernate,推荐使用 Envers;如果需要解耦业务逻辑,可以使用 AOP 或事件监听机制;如果希望与应用程序解耦,可以使用数据库触发器。
相关文章:
数据留痕的方法
在项目中,数据变更时,经常需要记录上次的数据,以便查看对比,专业术语叫做数据留痕。数据变更留痕(即记录数据的变更历史)是一个常见的需求,例如在审计、追踪数据变化或满足合规性要求的场景中。…...
机器学习数学基础:19.线性相关与线性无关
一、线性相关与线性无关的定义 (一)线性相关 想象我们有一组向量,就好比是一群有着不同“力量”和“方向”的小伙伴。给定的向量组 α ⃗ 1 , α ⃗ 2 , ⋯ , α ⃗ m \vec{\alpha}_1, \vec{\alpha}_2, \cdots, \vec{\alpha}_m α 1,α 2…...
ArgoCD实战指南:GitOps驱动下的Kubernetes自动化部署与Helm/Kustomize集成
摘要 ArgoCD 是一种 GitOps 持续交付工具,专为 Kubernetes 设计。它能够自动同步 Git 仓库中的声明性配置,并将其应用到 Kubernetes 集群中。本文将介绍 ArgoCD 的架构、安装步骤,以及如何结合 Helm 和 Kustomize 进行 Kubernetes 自动化部署。 引言 为什么选择 ArgoCD?…...
JVM虚拟机以及跨平台原理
相信大家已经了解到Java具有跨平台的特性,即“一次编译,到处运行”,例如在Windows下编写的程序,无需任何修改就可以在Linux下运行,这是C和C很难做到的。 那么,跨平台是怎样实现的呢?这就要谈及…...
【AIGC提示词系统】基于 DeepSeek R1 + ClaudeAI 易经占卜系统
上篇因为是VIP,这篇来一个免费的 提示词在最下方,喜欢的点个关注吧 引言 在人工智能与传统文化交融的今天,如何让AI系统能够传递传统易经文化的智慧,同时保持易经本身的神秘感和权威性,是一个极具挑战性的课题。本文将…...
电路笔记 : opa 运放失调电压失调电流输入偏置电流 + 反向放大器的平衡电阻 R3 = R1 // R2 以减小输出直流噪声
目录 定义影响和解决失调电压输入偏置电流平衡电阻R3推导公式: 失调电流 实际的运算放大器(Op-Amp)存在一些非理想特性,如失调电压(VIO)、失调电流(IIO)和输入偏置电流(I…...
ScrapeGraphAI颠覆传统网络爬虫技术
ScrapeGraphAI颠覆传统网络爬虫技术! 引言 在互联网时代,数据如同油田,丰富而深邃。但如何有效地提取这些数据,仍然是许多开发者面临的艰巨任务。你有没有想过,传统的网络爬虫技术是否已经过时?如今&…...
通过多层混合MTL结构提升股票市场预测的准确性,R²最高为0.98
“Boosting the Accuracy of Stock Market Prediction via Multi-Layer Hybrid MTL Structure” 论文地址:https://arxiv.org/pdf/2501.09760 摘要 本研究引入了一种创新的多层次混合多任务学习架构,致力于提升股市预测的效能。此架构融…...
java将list转成树结构
首先是实体类 public class DwdCusPtlSelectDto {//idprivate String key;//值private String value;//中文名private String title;private List<DwdCusPtlSelectDto> children;private String parentId;public void addChild(DwdCusPtlSelectDto child) {if(this.chil…...
互联网分布式ID解决方案
业界实现方案 1. 基于UUID 2. 基于DB数据库多种模式(自增主键、segment) 3. 基于Redis 4. 基于ZK、ETCD 5. 基于SnowFlake 6. 美团Leaf(DB-Segment、zkSnowFlake) 7. 百度uid-generator() 基于UUID生成唯一ID UUID生成策略 推荐阅读 DDD领域驱动与微服务架构设计设计模…...
xinference 安装(http导致错误解决)
为什么要使用xinference 安装xinference 环境 1)conda create -n Xinference python3.11 注意:3.9 3.10均可能出现xinference 安装时候出现numpy兼容性,以及无法安装all版本 错误: error while attempting to bind on address&am…...
334递增的三元子序列贪心算法(思路解析+源码)
文章目录 题目思路解析源码总结题目 思路解析 有两种解法:解法一:动态规划(利用dp找到数组最长递增序列长度,判断是否大于3即可)本题不适用,因为时间复杂度为O(n^2),超时。 解法二:贪心算法:解法如上图,题目要求长度为三,设置第一个元素为长度1的值,是指长度二的…...
【Linux】29.Linux 多线程(3)
文章目录 8.4 生产者消费者模型8.4.1 为何要使用生产者消费者模型8.4.2 生产者消费者模型优点 8.5 基于BlockingQueue的生产者消费者模型8.5.1 C queue模拟阻塞队列的生产消费模型 8.6. 为什么pthread_cond_wait 需要互斥量?8.7 条件变量使用规范8.8 条件变量的封装8.9 POSIX信…...
利用UNIAPP实现短视频上下滑动播放功能
在 UniApp 中实现一个短视频上下滑动播放的功能,可以使用 swiper 组件来实现滑动效果,并结合 video 组件来播放短视频。以下是一个完整的示例,展示如何在 UniApp 中实现这一功能。 1. 创建 UniApp 项目 如果你还没有创建 UniApp 项目,可以使用 HBuilderX 创建一个新的项目…...
vscode+CMake+Debug实现 及权限不足等诸多问题汇总
环境说明 有空再补充 直接贴两个json tasks.json {"version": "2.0.0","tasks": [{"label": "cmake","type": "shell","command": "cmake","args": ["../"…...
【提示词工程】探索大语言模型的参数设置:优化提示词交互的技巧
在与大语言模型(Large Language Model, LLM)进行交互时,提示词的设计和参数设置直接影响生成内容的质量和效果。无论是通过 API 调用还是直接使用模型,掌握模型的参数配置方法都至关重要。本文将为您详细解析常见的参数设置及其应用场景,帮助您更高效地利用大语言模型。 …...
基于 .NET 8.0 gRPC通讯架构设计讲解,客户端+服务端
目录 1.简要说明 2.服务端设计 2.1 服务端创建 2.2 服务端设计 2.3 服务端业务模块 3.客户端设计-控制台 4.客户端设计-Avalonia桌面程序 5.客户端设计-MAUI安卓端程序 1.简要说明 gRPC 一开始由 google 开发,是一款语言中立、平台中立、开源的远程过程调用…...
6.Centos7上部署flask+SQLAlchemy+python+达梦数据库
情况说明 前面已经介绍了window上使用pycharm工具开发项目时,window版的python连接达梦数据库需要的第三方包。 这篇文章讲述,centos7上的python版本连接达梦数据库需要的第三方包。 之前是在windows上安装达梦数据库的客户端,将驱动包安装到windows版本的python中。(开…...
【C语言系列】深入理解指针(5)
深入理解指针(5) 一、sizeof和strlen的对比1.1sizeof1.2strlen1.3sizeof和strlen的对比 二、数组和指针笔试题解析2.1 一维数组2.2 字符数组2.2.1代码1:2.2.2代码2:2.2.3代码3:2.2.4代码4:2.2.5代码5&#…...
mysql自连接 处理层次结构数据
MySQL 的自连接(Self Join)是一种特殊的连接方式,它允许一个表与自身进行连接。自连接通常用于处理具有层次结构或递归关系的数据,或者当同一张表中的数据需要相互关联时。以下是几种常见的场景,说明何时应该使用自连接…...
构建可靠AI编码代理:OpenClaw-Build工作流详解与实战
1. 项目概述:一个能“闭环”的AI编码代理工作流如果你用过市面上那些号称能自动编程的AI代理,大概率经历过这样的挫败感:你满怀期待地丢给它一个需求,它吭哧吭哧干了两三个任务,然后要么开始“神游”,写出来…...
自动化测试(十) 微服务测试策略-单元到集成到契约到端到端分层实战
微服务测试策略:单元→集成→契约→端到端分层实战前面咱们分别聊了单元测试、接口测试、契约测试。今天把它们串起来,聊聊微服务架构下怎么设计完整的测试策略——每一层测什么、怎么测、用什么工具。一、微服务测试的"金字塔"变体 单体应用的…...
为个人AI助手项目集成多模型API实现成本与性能平衡
🚀 告别海外账号与网络限制!稳定直连全球优质大模型,限时半价接入中。 👉 点击领取海量免费额度 为个人AI助手项目集成多模型API实现成本与性能平衡 构建个人AI助手是许多独立开发者热衷的项目。在开发过程中,一个常见…...
Windows平台iOS模拟器开发革命:ipasim如何让iOS应用在Windows上“原生“运行
Windows平台iOS模拟器开发革命:ipasim如何让iOS应用在Windows上"原生"运行 【免费下载链接】ipasim iOS emulator for Windows 项目地址: https://gitcode.com/gh_mirrors/ip/ipasim 嘿,开发者朋友们!你是否曾经梦想过在Win…...
3个步骤搭建Sunshine游戏串流服务器:从零到一的完整指南
3个步骤搭建Sunshine游戏串流服务器:从零到一的完整指南 【免费下载链接】Sunshine Self-hosted game stream host for Moonlight. 项目地址: https://gitcode.com/GitHub_Trending/su/Sunshine 你是否曾经梦想过在客厅的电视上玩书房电脑里的3A大作…...
构建离线优先应用终极指南:Material Components Web 与 Service Worker 完美集成
构建离线优先应用终极指南:Material Components Web 与 Service Worker 完美集成 【免费下载链接】material-components-web Modular and customizable Material Design UI components for the web 项目地址: https://gitcode.com/gh_mirrors/ma/material-compone…...
从零部署Discord AI聊天机器人:基于ChatGPT API与Firestore的实践指南
1. 项目概述:打造一个属于你自己的Discord AI聊天机器人 如果你在运营一个Discord社区,无论是游戏公会、技术讨论组还是兴趣社团,肯定遇到过这样的场景:成员们总有一些稀奇古怪的问题,或者需要一个随时在线的“智能助…...
MMC柔性直流输电稳定性与参数控制【附代码】
✨ 长期致力于模块化多电平换流器、弱交流电网、小信号模型、控制器参数优化、粒子群算法、模糊控制研究工作,擅长数据搜集与处理、建模仿真、程序编写、仿真设计。 ✅ 专业定制毕设、代码 ✅ 如需沟通交流,点击《获取方式》 (1)弱…...
基于VLLM与VoxCPM2的高并发TTS服务器部署与调优指南
1. 项目概述:uttera-tts-vllm,一个为高并发而生的TTS服务器如果你正在寻找一个能扛住高并发请求、支持实时语音克隆、并且完全自托管的文本转语音解决方案,那么uttera-tts-vllm绝对值得你花时间研究一下。这个项目本质上是一个基于 FastAPI 构…...
手机号查QQ号终极指南:3分钟掌握Python逆向查询技巧
手机号查QQ号终极指南:3分钟掌握Python逆向查询技巧 【免费下载链接】phone2qq 项目地址: https://gitcode.com/gh_mirrors/ph/phone2qq 你是否曾需要快速验证手机号与QQ号的绑定关系?手机号查QQ号工具是一个简单高效的Python开源项目࿰…...
