当前位置:首页 > 文章列表 > 数据库 > MySQL > MySQL 审计日志小项目:记录订单状态变更、查询和清理

MySQL 审计日志小项目:记录订单状态变更、查询和清理

来源:17golang原创 2026-06-29 18:03:25 0浏览 收藏

订单状态从“待支付”变成“已支付”,再变成“已发货”,后台通常只保留当前状态。等到客服追问“谁在什么时候改了状态”,或者运营发现订单状态异常时,如果没有审计日志,就只能翻业务日志,定位成本很高。

本文做一个小项目:用 MySQL 记录订单状态变更审计日志。项目包含订单表、审计表、触发器、查询 SQL 和清理策略。目标不是把所有审计能力一次做完,而是先搭一个可运行、可查询、可维护的基础版本。

目录
  • 项目目标:记录每一次订单状态变化
  • 环境准备:创建订单表和审计表
  • 核心代码:用触发器写入状态变更日志
  • 本地运行:插入订单并模拟状态流转
  • 部署集成:业务代码和触发器怎么配合
  • 验收清单:查询、索引和过期清理

项目目标:记录每一次订单状态变化

这个小项目要解决三个问题:

  • 订单状态变更后,能看到旧状态、新状态、变更时间。
  • 按订单号查询时,能按时间线看到完整状态流转。
  • 审计日志长期增长后,可以按时间分批清理。

数据生命周期可以理解为:业务更新订单状态,MySQL 触发审计记录,审计表追加一行日志,查询时按订单聚合时间线,最后按保留周期清理历史数据。

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

MySQL 审计日志本地运行流程:插入订单、更新状态、触发器写入、查询时间线、清理历史

部署集成:业务代码和触发器怎么配合

触发器适合做兜底,但业务代码仍然要负责提供清晰的操作者和变更时间。例如支付回调更新状态时,把 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 审计日志小项目的核心是把“当前订单状态”和“历史状态变化”分开存储。订单表保存当前值,审计表追加记录每一次变化;触发器或业务代码负责在同一事务里写入审计;查询时按订单时间线展示;清理时按时间分批处理。这样既能满足排查和追溯,又不会把审计逻辑散落到各个页面里。

版本声明
本文转载于:17golang原创 如有侵犯,请联系study_golang@163.com删除
Java Webhook 回调接收接口设计:验签、幂等和状态流转Java Webhook 回调接收接口设计:验签、幂等和状态流转
上一篇
Java Webhook 回调接收接口设计:验签、幂等和状态流转
PHP-FPM 慢请求报警运行手册:从 slowlog 到进程池参数调整
下一篇
PHP-FPM 慢请求报警运行手册:从 slowlog 到进程池参数调整
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    516次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    500次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    485次学习
查看更多
AI推荐
  • ljg-skills -
    ljg-skills
    ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
    2903次使用
  • MELO音乐 - AI 音乐生成平台,支持多模态创作能力
    MELO音乐
    MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
    2691次使用
  • UniScribe - AI 免费在线音视频转文字平台
    UniScribe
    UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
    2624次使用
  • 剧云 - 免费 AI 智能中文剧本创作平台
    剧云
    剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
    2857次使用
  • 万象有声 - AI 一站式有声内容创作平台
    万象有声
    万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
    2797次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码