MySQL中的事件调度基础学习教程 |
经常需要有一些定时任务在MySQL表上执行,例如统计、迁移、删除无用数据等 。之前的作法是利用Linux cron定时运行脚本,但是发现这样的额外依赖有时并不方便,例如单机多实例部署时,就需要分别手动分别配置不同的cron任务,需要额外配置相应的用户和权限;新环境部署时容易遗漏cron任务等 。 MySQL提供了Event Scheduler,与Linux下的crontab类似,可以根据时间调度来运行任务,运行一次或多次 。 完整的Event Schduler创建语句如下: CREATE [DEFINER = { user | CURRENT_USER }] EVENT [IF NOT EXISTS] event_name ON SCHEDULE schedule [ON COMPLETION [NOT] PRESERVE] [ENABLE | DISABLE | DISABLE ON SLAVE] [COMMENT comment] DO event_body; 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} 一、调度Scheduler ON SCHEDULE AT timestamp [+ INTERVAL interval] … | EVERY interval [STARTS timestamp [+ INTERVAL interval] …] [ENDS timestamp [+ INTERVAL interval] …] 其中,timestamp必须包括”年月日时分秒“,它参与表达式计算后,结果是datetime或者timestamp类型 。 而时间间隔interval可以如下: <数字> {YEAR | QUARTER | MONTH | DAY | HOUR | MINUTE | WEEK | SECOND | YEAR_MONTH | DAY_HOUR | DAY_MINUTE | DAY_SECOND | HOUR_MINUTE | HOUR_SECOND | MINUTE_SECOND} 其含义很清晰,如YEAR 年;QUARTER 季度;YEAR_MONTH 年+月;MINUTE_SECOND 分钟+秒 。 补充: YEAR | QUARTER | MONTH | YEAR_MONTH 后台都转换成MONTH,其他时间间隔都转换成SECOND 示例: AT 2006-02-10 23:59:00′ 指定确切运行时间,本地时区 。 指定STARTS、ENDS是可选的 。 STARTS是指定重复运行的第一次是什么时候 。不指定的情况下,会在事件创建时运行第一次,即等价于STARTS CURRENT_TIMESTAMP! EVERY 5 WEEK 每5周运行一次,创建时运行第一次 。 7384313 event_scheduler localhost [NULL] Daemon 3 Waiting on empty queue [NULL] 默认事件调度Event Scheduler功能是未启用的,需要配置全局参数event_scheduler,本参数可以动态设置,即时生效 。 event_scheduler有如下三种取值: OFF/0 关闭,默认值 。不运行Event Scheduler线程,也就无法进行事件调度 。设置为ON可以立即启用 。 set global event_scheduler=on 要随MySQL服务一起启用,则在/etc/my.cnf中添加 [mysqld] event-scheduler=on 2.创建事件的语法 CREATE [DEFINER = { user | CURRENT_USER }] EVENT [IF NOT EXISTS] event_name ON SCHEDULE schedule [ON COMPLETION [NOT] PRESERVE] [ENABLE | DISABLE | DISABLE ON SLAVE] [COMMENT comment] DO event_body; 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} 参数详细说明: ALTER [DEFINER = { user | CURRENT_USER }] EVENT event_name [ON SCHEDULE schedule] [ON COMPLETION [NOT] PRESERVE] [RENAME TO new_event_name] [ENABLE | DISABLE | DISABLE ON SLAVE] [COMMENT comment] [DO event_body] 4.删除事件的语法 DROP EVENT [IF EXISTS] event_name 5.Do子句 1)简单SQL示例: CREATE EVENT e_hourly ON SCHEDULE EVERY 1 HOUR COMMENT ‘Clears out sessions table each hour. DO DELETE FROM site_activity.sessions; 2)复杂SQL示例: delimiter | CREATE EVENT e ON SCHEDULE EVERY 5 SECOND DO BEGIN DECLARE v INTEGER; DECLARE CONTINUE HANDLER FOR SQLEXCEPTION BEGIN END; SET v = 0; WHILE v < 5 DO INSERT INTO t1 VALUES (0); UPDATE t2 SET s1 = s1 + 1; SET v = v + 1; END WHILE; END | delimiter ; 3)Do子句中SQL的限制 基本上Do中可以使用任何在存储程序(Stored Routine)中允许的SQL语句,而存储程序中有些限制,event还有些额外的限制 。 Stored Routine中如下语句不允许:
支持动态SQL(PREPARE, EXECUTE, DEAALOCATE PREPARE)!但是PREPARE本身有些语句不允许执行 。 INSERT DELAYED不会生效 如果Do子句中包含ALTER EVENT子句,虽然能够创建,但是运行时会出错 。
mysql.event information_schema.events show events show create event
四、 Mysql事件实战 CREATE TABLE `test` ( `id` int(11) NOT NULL AUTO_INCREMENT, `t1` datetime DEFAULT NULL, `id2` int(11) NOT NULL DEFAULT 0, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=106 DEFAULT CHARSET=utf8 实战1 CREATE EVENT IF NOT EXISTS test ON SCHEDULE EVERY 3 SECOND ON COMPLETION PRESERVE DO INSERT INTO test(id,t1) VALUES(,NOW()); Ø 创建一个10分钟后清空test表数据的事件 CREATE EVENT IF NOT EXISTS test ON SCHEDULE AT CURRENT_TIMESTAMP + INTERVAL 1 MINUTE DO TRUNCATE TABLE test.aaa; Ø 创建一个在2012-08-23 00:00:00时刻清空test表数据的事件,代码如下: CREATE EVENT IF NOT EXISTS test ON SCHEDULE AT TIMESTAMP 2012-08-23 00:00:00 DO TRUNCATE TABLE test; Ø 创建一个从2012年8月22日21点45分开始到10分钟后结束,运行每隔3秒往test表中插入一条数据的事件,代码如下: CREATE EVENT IF NOT EXISTS test ON SCHEDULE EVERY 3 SECOND STARTS 2012-08-22 21:49:00 ENDS 2012-08-22 21:49:00+ INTERVAL 10 MINUTE ON COMPLETION PRESERVE DO INSERT INTO test(id,t1) VALUES(,NOW()); DROP PROCEDURE IF EXISTS test_add; DELIMITER // CREATE PROCEDURE test_add() BEGIN DECLARE 1_id INT DEFAULT 1; DECLARE 1_id2 INT DEFAULT 0; DECLARE error_status INT DEFAULT 0; DECLARE datas CURSOR FOR SELECT id FROM test; DECLARE CONTINUE HANDLER FOR NOT FOUND SET error_status=1; OPEN datas; FETCH datas INTO 1_id; REPEAT SET 1_id2=1_id2+2; UPDATE test SET id2=1_id2 WHERE id=1_id; FETCH datas INTO 1_id; UNTIL error_status END REPEAT; CLOSE datas; END // 事件设置2012-08-22 00:00:00时刻开始运行,每隔1调用一次存储过程,40天后结束,代码如下: CREATE EVENT test ON SCHEDULE EVERY 1 DAY STARTS 2012-08-22 00:00:00 ENDS 2012-08-22 00:00:00+INTERVAL 40 DAY ON COMPLETION PRESERVE DO CALL test_add(); |