MySQL 定时器实战:从入门到进阶
作为一名后端开发者,你是否遇到过这些场景:电商系统需要每天凌晨自动清理过期的临时订单,论坛系统需要每周生成一次用户活跃度报表,日志系统需要每月归档一次历史数据?这些周期性任务如果通过应用层定时任务实现,难免会面临服务器重启、分布式任务调度等问题。其实 MySQL 自带的定时器(Event Scheduler)就能优雅解决这些问题,今天就带大家深入了解这个宝藏功能。
MySQL 从 5.1 版本开始引入 Event Scheduler 功能,通过创建事件(Event)实现定时执行 SQL 语句的需求。不同于触发器(Trigger)的实时响应,事件支持按计划执行,支持单次执行和循环执行,甚至能精确到秒级调度。
首先需要确认服务器是否启用了事件调度器:
-- 查看全局状态
SHOW VARIABLES LIKE 'event_scheduler';
-- 输出结果为ON则已启用,OFF则需要开启
SET GLOBAL event_scheduler = ON;
生产环境建议在 my.cnf 配置文件中添加event_scheduler = ON,避免重启后失效。
事件创建语法遵循:
12345678910111213 | CREATE EVENT [IF NOT EXISTS] event_nameON SCHEDULE schedule[ON COMPLETION [NOT] PRESERVE][ENABLE | DISABLE | DISABLE ON SLAVE]DO event_body;-- 其中schedule支持多种时间格式schedule ::=AT timestamp [+ INTERVAL interval] ...| EVERY interval [STARTS timestamp [+ INTERVAL interval] ...][ENDS timestamp [+ INTERVAL interval] ...]interval ::=quantity { YEAR | QUARTER | MONTH | DAY | HOUR | MINUTE |WEEK | SECOND | YEAR_MONTH | DAY_HOUR | DAY_MINUTE |DAY_SECOND | HOUR_MINUTE | HOUR_SECOND | MINUTE_SECOND } |
---|
场景:用户下单后 30 分钟未支付,自动取消订单并释放库存
-- 创建事件
DELIMITER $$CREATE EVENT
cancel_unpaid_orders
ON SCHEDULE EVERY 1 MINUTESTARTS
CURRENT_TIMESTAMPDOBEGINUPDATE orders
SET status = 'canceled',
update_time = NOW()WHERE
status = 'pending'AND create_time < DATE_SUB(NOW(),
INTERVAL 30 MINUTE);
-- 释放库存(假设库存表通过订单ID关联)
UPDATE order_items oi
INNER JOIN
products p ON
oi.product_id = p.idSET
p.stock = p.stock + oi.quantityWHERE
oi.order_id IN (SELECT id FROM orders
WHERE
status = 'canceled'AND update_time = NOW());END$$DELIMITER ;
-- 查看事件状态
SHOW EVENTS LIKE 'cancel_unpaid_orders';
场景:每月 1 号凌晨 2 点将上个月的操作日志归档到历史表
-- 创建一次性事件
CREATE EVENT archive_operation_logs
ON SCHEDULE AT '2023-10-01 02:00:00'DOBEGIN
-- 创建归档表(如果不存在)
SET @archive_table = CONCAT('operation_log_', DATE_FORMAT(CURRENT_DATE - INTERVAL 1 MONTH, '%Y%m'));
SET @create_sql = CONCAT('CREATE TABLE IF NOT EXISTS ', @archive_table, ' LIKE operation_logs');
PREPARE stmt FROM @create_sql;EXECUTE stmt;DEALLOCATE PREPARE stmt;
-- 移动数据
INSERT INTO `operation_logs_202309`
(SELECT * FROM operation_logs WHERE create_time < '2023-10-01');
DELETE FROM operation_logs WHERE create_time < '2023-10-01';END;
-- 设置事件执行后自动删除(默认会保留事件定义)
ALTER EVENT archive_operation_logs ON COMPLETION NOT PRESERVE;
场景:保留最近 30 天的慢查询日志,每天凌晨清理过期数据
CREATE EVENT clean_slow_query_logsON
SCHEDULE EVERY 1 DAYSTARTS '2023-09-01 00:00:00'DOBEGINDELETE FROM slow_query_logsWHERE create_time < DATE_SUB(NOW(), INTERVAL 30 DAY);-- 可选:添加重试机制(处理删除锁表问题)IF ROW_COUNT() < 0 THENRESIGNAL; -- 抛出异常,可结合错误处理机制END IF;END;
DELIMITER $$CREATE EVENT safe_transaction_eventON SCHEDULE EVERY 1 HOURDOBEGINSTART TRANSACTION;UPDATE user_balance SET balance = balance - 100 WHERE user_id = 1;-- 模拟可能失败的操作INSERT INTO transaction_record (user_id, amount) VALUES (1, -100);COMMIT;-- 错误处理(需在存储过程中使用)DECLARE EXIT HANDLER FOR SQLEXCEPTIONBEGINROLLBACK;INSERT INTO event_error_log VALUES (NOW(), 'safe_transaction_event', ERROR_MESSAGE());END;END$$DELIMITER ;
当表名或时间参数需要动态生成时,可使用预处理语句:
SET @table_name = CONCAT('report_', DATE_FORMAT(NOW(), '%Y%m'));
SET @sql = CONCAT('CREATE TABLE IF NOT EXISTS ', @table_name, ' (id INT PRIMARY KEY)');
PREPARE stmt FROM @sql;EXECUTE stmt;DEALLOCATE PREPARE stmt;
事件执行时使用的是创建者的权限,建议为事件单独创建专用账户:
-- 创建事件专用用户
CREATE USER 'event_user'@'%' IDENTIFIED BY 'StrongPassword123!';
GRANT EVENT, INSERT, UPDATE, DELETE ON mydb.* TO 'event_user'@'%';
-- 指定事件使用该用户执行(需超级权限)
ALTER EVENT event_name DISABLE;
-- 先禁用事件
EXNESS官方网站为:https://one.exnesstrack.org/a/l03oed5fra
ALTER EVENT event_name OWNER TO 'event_user'@'%';ALTER EVENT event_name ENABLE;
DELETE FROM old_data LIMIT 1000;
事件调度使用的是服务器时区,建议通过SET GLOBAL time_zone = '+8:00';统一时区,或在时间参数中使用 UTC 时间
开启事件调度器日志(需修改 my.cnf):
[mysqld]log_events =
ONlog_output =
FILEevent_scheduler_log_file =
/var/log/mysql/event_scheduler.log
-- 查看所有事件
SHOW EVENTS;
-- 查看事件定义
SHOW CREATE EVENT event_name;
-- 临时禁用事件(不删除定义)
ALTER EVENT event_name DISABLE;
-- 重新启用事件
ALTER EVENT event_name ENABLE;
-- 修改执行时间(无需重建事件)
ALTER EVENT event_name ON SCHEDULE EVERY 2 HOUR;
-- 删除事件
DROP EVENT IF EXISTS event_name;
建议建立事件管理表,记录每个事件的用途、负责人、执行频率、最后执行时间等信息,方便后续维护:
CREATE TABLE event_management
(event_name VARCHAR(64)
PRIMARY KEY,description TEXT,
owner VARCHAR(32),
schedule_rule VARCHAR(128),
last_execute_time TIMESTAMP,
create_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP);
场景特征 | 适合使用事件调度器 | 更适合应用层定时任务 |
---|---|---|
纯数据库操作 | ✅ | ❌ |
跨语言 / 跨平台 | ❌ | ✅ |
高频率小规模任务 | ✅(<1 分钟间隔) | ❌(资源消耗大) |
复杂业务逻辑 | ❌(需存储过程) | ✅ |
分布式一致性要求 | ✅(事务支持) | ❌(需额外协调) |
MySQL 定时器作为数据库层的调度工具,在处理纯 SQL 的周期性任务时具有得天独厚的优势。合理运用事件调度器,既能减轻应用层压力,又能保证数据操作的原子性。下次遇到需要定时执行的数据库任务时,不妨试试这个内置的强大功能吧!
记得在使用前评估任务的复杂度和性能影响,生产环境建议先在测试库充分验证,并且做好事件执行失败的监控和重试机制。
本文系转载,前往查看
如有侵权,请联系 cloudcommunity@tencent.com 删除。
本文系转载,前往查看
如有侵权,请联系 cloudcommunity@tencent.com 删除。