触发器中禁止耗时操作,应改用异步方案:MySQL用消息表+轮询,PostgreSQL优先用LISTEN/NOTIFY;需保障幂等、唯一ID、上下文完整及超时重试。触发器里直接调用耗时操作必然拖垮事务SQL 触发器运行在主事务上下文中,INSERT/UPDATE/DELETE 不完成,触发器不返回,整个事务就卡着。哪怕只是多查一次远程 API、写日志到慢盘、或发一封邮件,都可能让 LOCK WAIT TIMEOUT EXCEEDED 或 Transaction timeout 频繁报出。常见错误现象:业务接口偶发 504,数据库监控显示 innodb_row_lock_time_avg 突增,慢查询日志里看不到大 SQL,但事务等待时间明显拉长。触发器中禁止出现任何网络 I/O(如 curl、http_request)、文件写入、跨库查询避免在触发器中调用存储过程,除非该过程已明确验证为纯内存计算、无锁、毫秒级完成MySQL 8.0+ 的 VALIDATE PASSWORD 类插件函数也需警惕——某些配置下会隐式触发磁盘读用「插入消息表 + 定时轮询」替代同步触发逻辑最轻量、兼容性最强的解法:把原本想在触发器里干的事,改成往一张专用消息表里插一条记录,再由外部进程异步消费。不改应用代码,不依赖消息中间件,MySQL 原生支持。使用场景:审计日志落 ES、订单变更通知下游系统、库存扣减后更新缓存等——只要不要求“实时”,只要求“最终一致”。建表:CREATE TABLE trigger_queue (id BIGINT AUTO_INCREMENT PRIMARY KEY, event_type VARCHAR(32), payload JSON, created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP)原触发器只保留:INSERT INTO trigger_queue (event_type, payload) VALUES ('order_updated', JSON_OBJECT('order_id', NEW.id, 'status', NEW.status))外部消费者用 SELECT ... FOR UPDATE SKIP LOCKED 安全取任务,处理完再 DELETE,避免重复消费PostgreSQL 用户优先考虑 LISTEN/NOTIFY + worker 进程比起轮询,LISTEN/NOTIFY 是 PostgreSQL 原生的轻量事件通知机制,无轮询延迟、无额外表压力、事务提交即触发。 唱鸭 音乐创作全流程的AI自动作曲工具,集 AI 辅助作词、AI 自动作曲、编曲、混音于一体
如何防止SQL触发器导致事务超时_拆分逻辑为异步队列处理