MySQL 审计日志小项目:记录订单状态变更、查询和清理
订单状态从“待支付”变成“已支付”,再变成“已发货”,后台通常只保留当前状态。等到客服追问“谁在什么时候改了状态”,或者运营发现订单状态异常时,如果没有审计日志,就只能翻业务日志,定位成本很高。
本文做一个小项目:用 MySQL 记录订单状态变更审计日志。项目包含订单表、审计表、触发器、查询 SQL 和清理策略。目标不是把所有审计能力一次做完,而是先搭一个可运行、可查询、可维护的基础版本。
- 项目目标:记录每一次订单状态变化
- 环境准备:创建订单表和审计表
- 核心代码:用触发器写入状态变更日志
- 本地运行:插入订单并模拟状态流转
- 部署集成:业务代码和触发器怎么配合
- 验收清单:查询、索引和过期清理
项目目标:记录每一次订单状态变化
这个小项目要解决三个问题:
- 订单状态变更后,能看到旧状态、新状态、变更时间。
- 按订单号查询时,能按时间线看到完整状态流转。
- 审计日志长期增长后,可以按时间分批清理。
数据生命周期可以理解为:业务更新订单状态,MySQL 触发审计记录,审计表追加一行日志,查询时按订单聚合时间线,最后按保留周期清理历史数据。

审计日志要尽量保持追加写,不要覆盖。因为它记录的是“发生过什么”,不是当前状态。
环境准备:创建订单表和审计表
先准备一张简化订单表。真实项目里字段会更多,这里只保留演示需要的字段。
CREATE TABLE orders ( id BIGINT PRIMARY KEY AUTO_INCREMENT, order_no VARCHAR(64) NOT NULL UNIQUE, user_id BIGINT NOT NULL, status VARCHAR(32) NOT NULL, amount DECIMAL(12,2) NOT NULL, updated_by VARCHAR(64) NOT NULL, created_at DATETIME NOT NULL, updated_at DATETIME NOT NULL );
审计表单独保存状态变化。这里把旧状态、新状态、操作者和变更时间都放进去。
CREATE TABLE order_status_audit ( id BIGINT PRIMARY KEY AUTO_INCREMENT, order_id BIGINT NOT NULL, order_no VARCHAR(64) NOT NULL, old_status VARCHAR(32) NOT NULL, new_status VARCHAR(32) NOT NULL, changed_by VARCHAR(64) NOT NULL, changed_at DATETIME NOT NULL, created_at DATETIME NOT NULL, KEY idx_order_time (order_id, changed_at), KEY idx_changed_at (changed_at) );
idx_order_time 用于按订单查时间线,idx_changed_at 用于后续按时间清理。审计表的查询和清理路径一开始就要考虑,否则数据量上来后会很难补。
核心代码:用触发器写入状态变更日志
状态变更可以由业务代码主动写审计表,也可以用触发器兜底。这个小项目使用触发器,重点演示“只要订单状态发生变化,就追加审计记录”。
DELIMITER //
CREATE TRIGGER trg_orders_status_audit
AFTER UPDATE ON orders
FOR EACH ROW
BEGIN
IF OLD.status NEW.status THEN
INSERT INTO order_status_audit (
order_id,
order_no,
old_status,
new_status,
changed_by,
changed_at,
created_at
) VALUES (
NEW.id,
NEW.order_no,
OLD.status,
NEW.status,
NEW.updated_by,
NEW.updated_at,
NOW()
);
END IF;
END//
DELIMITER ;
这里有一个关键判断:只有 OLD.status NEW.status 时才写审计日志。如果只是修改金额、备注或更新时间,不应该生成状态变更审计。
本地运行:插入订单并模拟状态流转
先插入一条待支付订单:
INSERT INTO orders ( order_no, user_id, status, amount, updated_by, created_at, updated_at ) VALUES ( 'SO202606290001', 10001, 'WAIT_PAY', 199.00, 'system', NOW(), NOW() );
再模拟两次状态变化:
UPDATE orders
SET status = 'PAID',
updated_by = 'pay-callback',
updated_at = NOW()
WHERE order_no = 'SO202606290001';
UPDATE orders
SET status = 'SHIPPED',
updated_by = 'warehouse-admin',
updated_at = NOW()
WHERE order_no = 'SO202606290001';
查询审计时间线:
SELECT order_no, old_status, new_status, changed_by, changed_at FROM order_status_audit WHERE order_no = 'SO202606290001' ORDER BY changed_at ASC;
预期可以看到两行记录:WAIT_PAY -> PAID,以及 PAID -> SHIPPED。

部署集成:业务代码和触发器怎么配合
触发器适合做兜底,但业务代码仍然要负责提供清晰的操作者和变更时间。例如支付回调更新状态时,把 updated_by 写成 pay-callback;后台人工发货时,把 updated_by 写成人员账号或系统账号。
集成时建议遵守三条规则:
- 订单状态只能通过统一入口更新,避免绕过
updated_by。 - 触发器只记录事实,不写复杂业务判断。
- 审计表只追加,不在业务流程里修改历史审计行。
如果系统对触发器比较敏感,也可以改成业务代码主动写审计表。但无论哪种方式,都要保持同一条原则:状态变更和审计记录要在同一个事务里完成。
验收清单:查询、索引和过期清理
上线前先按下面清单验收。
按订单查询是否稳定
用 EXPLAIN 看是否命中 idx_order_time:
EXPLAIN SELECT old_status, new_status, changed_by, changed_at FROM order_status_audit WHERE order_id = 1 ORDER BY changed_at ASC;
重复更新是否会产生无效日志
如果状态没有变化,只更新 updated_at,审计表不应该新增记录。
历史数据是否能分批清理
审计日志通常要保留一段时间,例如 180 天。清理时不要一次删除太多行,建议分批处理。
DELETE FROM order_status_audit WHERE changed_at
清理任务跑完后再继续下一批,避免长事务和锁等待。对审计要求更高的业务,可以先归档到冷库,再删除在线表数据。
总结
这个 MySQL 审计日志小项目的核心是把“当前订单状态”和“历史状态变化”分开存储。订单表保存当前值,审计表追加记录每一次变化;触发器或业务代码负责在同一事务里写入审计;查询时按订单时间线展示;清理时按时间分批处理。这样既能满足排查和追溯,又不会把审计逻辑散落到各个页面里。
Java Webhook 回调接收接口设计:验签、幂等和状态流转
- 上一篇
- Java Webhook 回调接收接口设计:验签、幂等和状态流转
- 下一篇
- PHP-FPM 慢请求报警运行手册:从 slowlog 到进程池参数调整
-
- 数据库 · MySQL | 2星期前 | MySQL · 慢查询 · 索引优化 · COUNT查询 · 汇总表 · 联合索引 覆盖索引 汇总表 MySQL COUNT慢 COUNT(*)优化
- MySQL COUNT(*) 总数查询变慢怎么办:从扫描行数到汇总表的完整治理流程
- 329浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 2903次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 2691次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 2624次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 2857次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 2797次使用
-
- Linux系统下如何安装Mysql(centOS7以上不支持Mysql)
- 2023-01-16 100浏览
-
- 在windows上用docker desktop安装StoneDB
- 2023-01-20 100浏览
-
- 总结 mysql 一些小技巧
- 2023-01-21 100浏览
-
- MySQL如何给大表加索引
- 2023-01-26 100浏览
-
- 积分商城简要设计
- 2023-02-17 100浏览

