SQL 实时任务调度核心靠数据库原生事件(定时执行)和触发器(数据变动即响应),二者配合覆盖多数场景:事件需开启 event_scheduler,支持 EVERY/STARTS/ENDS;触发器分 BEFORE/AFTER,通过 NEW/OLD 访问数据;组合应用如触发器写缓存、事件批量处理;需注意权限、性能与稳定性。

SQL 实时任务调度不靠外部脚本,核心是用好数据库原生的事件(Event)和触发器(Trigger)两类机制。事件解决“定时执行”,触发器解决“数据一变就响应”,两者分工明确,配合使用才能覆盖大多数实时场景。
MySQL 事件:精准到秒的定时动作
MySQL 事件调度器(event_scheduler)必须先开启,否则所有事件都不生效。检查命令:
SHOW VARIABLES LIKE ‘event_scheduler’;
返回 ON 才算启用成功。临时开启用:
SET GLOBAL event_scheduler = ON;
永久生效需在 my.cnf 里加 event_scheduler=ON 并重启服务。
创建一个每 5 分钟清理一次日志表的事件示例:
- 语法结构清晰:EVENT 名称 + ONSCHEDULE 定义时间 + DO 包裹实际操作
- 时间表达灵活:可用 EVERY 5 MINUTE,也可加 STARTS 和 ENDS 控制生命周期
- DO 块中支持多条语句,但不能有 SELECT 返回结果集(可改用 INSERT … SELECT 或写入日志表)
- 事件默认启用,如需暂停可加 DISABLE 关键字
触发器:数据变动即刻响应
触发器不是定时运行,而是绑定在表的 DML 操作上,只要 INSERT/UPDATE/DELETE 发生,立刻触发。它没有时间概念,只认“事件发生”。
关键细节:
- BEFORE 类型适合校验或改写待插入 / 更新的数据(比如自动补全 create_time、修正负数 salary)
- AFTER 类型适合记录日志、更新统计表、触发下游同步(如订单状态变更后更新订单明细表)
- 访问变更前后数据靠 NEW(新值)和 OLD(旧值),仅限行级触发器(FOR EACH ROW)
- 不能调用存储过程以外的外部服务,逻辑应尽量轻量,避免阻塞主事务
事件与触发器如何配合用
单独用事件只能做周期性维护,单独用触发器无法处理“每天凌晨汇总”这类时间驱动任务。真实业务常组合使用:
- 用触发器捕获每笔订单创建,实时写入中间缓存表
- 再用事件每 10 秒扫描缓存表,把积攒的订单批量落库或推送到消息队列
- 用触发器拦截非法 UPDATE,同时用事件定期检查异常数据并告警
- 审计类需求:触发器记操作日志,事件每天凌晨压缩归档旧日志
注意权限与稳定性边界
事件需要用户具备 EVENT 权限;触发器需要 TRIGGER 权限;两者执行内部 SQL 还需对应表的 INSERT/UPDATE 等权限。
稳定性提醒:
- 事件调度精度可达秒级,但高并发下大量事件同时触发可能争抢资源
- 触发器逻辑别写太重,尤其避免在 BEFORE UPDATE 中调用复杂子查询或远程接口
- 事件不支持手动调用,测试时建议先用存储过程封装逻辑,再由事件调用
- 生产环境事件建议加上 ON COMPLETION PRESERVE,便于排查和复用
基本上就这些。用对机制,SQL 层就能扛起不少原本要靠应用层或调度平台完成的实时任务。