bitscn.com
自mysql5.1.6起,增加了一个非常有特色的功能–事件调度器(event scheduler),可以用做定时执行某些特定任务(例如:删除记录、对数据进行汇总等等),来取代原先只能由操作系统的计划任务来执行的工作。更值得一提的是mysql的事件调度器可以精确到每秒钟执行一个任务,而操作系统的计划任务(如:linux下的cron或windows下的任务计划)只能精确到每分钟执行一次。对于一些对数据实时性要求比较高的应用(例如:股票、赔率、比分等)就非常适合。
事件调度器有时也可称为临时触发器(temporal triggers),因为事件调度器是基于特定时间周期触发来执行某些任务,而触发器(triggers)是基于某个表所产生的事件触发的,区别也就在这里。
在使用这个功能之前必须确保event_scheduler已开启,可执行
set global event_scheduler = 1;
或我们可以在配置my.ini文件 中加上event_scheduler = 1或
set global event_scheduler = on;
来开启,也可以直接在启动命令加上“–event_scheduler=1”,例如:
mysqld ... --event_scheduler=1
要查看当前是否已开启事件调度器,可执行如下sql:
show variables like 'event_scheduler';
或
select @@event_scheduler;
或
拥有super 权限的账户执行show processlist 就可以看到这个线程了
5.3、定时服务配置
先来看一下它的语法:
create event [if not exists] event_name
on schedule schedule
[on completion [not] preserve]
[enable | disable]
[comment 'comment']
do sql_statement;
schedule:
at timestamp [+ interval interval]
| every interval [starts timestamp] [ends timestamp]
interval:
quantity {year | quarter | month | day | hour | minute |
week | second | year_month | day_hour | day_minute |
day_second | hour_minute | hour_second | minute_second}
5.3.1每秒插入一条记录到数据表
use test;
create table aaa (timeline timestamp);
create event e_test_insert
on schedule every 1 second
do insert into test.aaa values (current_timestamp);
等待3秒钟后,再执行查询看看:
mysql> select * from aaa;
+---------------------+
| timeline |
+---------------------+
| 2007-07-18 20:44:26 |
| 2007-07-18 20:44:27 |
| 2007-07-18 20:44:28 |
+---------------------+
5.3.2 5秒(天)后清空test表
create event e_test
on schedule at current_timestamp + interval 5 second
do truncate table test.aaa;
create event e_test
on schedule at current_timestamp + interval 5 day
do truncate table test.aaa;
5.3.3 2008年5月23日9点39分20秒整清空test表
create event e_test
on schedule at timestamp '2008-05-23 9:39:20'
do truncate table test.aaa;
这个测试有问题。还不太明白原因。
5.3.4每天定时清空test表
create event e_test
on schedule every 1 day
do truncate table test.aaa;
5.3.5 5天后开启每天定时清空test表
create event e_test
on schedule every 1 day
starts current_timestamp + interval 5 day
do truncate table test.aaa;
这里5天也可以为0天,当时就开启清空表
5.3.6每天定时清空test表,5天后停止执行
create event e_test
on schedule every 1 day
ends current_timestamp + interval 5 day
do truncate table test.aaa;
该设置要求天数大于1,否则报错。而且创建不成功
5.3.7 5天后开启每天定时清空test表,一个月后停止执行
create event e_test
on schedule every 1 day
starts current_timestamp + interval 5 day
ends current_timestamp + interval 1 month
do truncate table test.aaa;[on completion [not] preserve]
可以设置这个事件是执行一次还是持久执行,默认为not preserve。
该事件会停止每隔一秒插入数据的事件,感觉这点上mysql做的还是有问题。
5.3.8每天定时清空test表(只执行一次,任务完成后就终止该事件)
create event e_test
on schedule every 1 day
on completion not preserve
do truncate table test.aaa;
[enable | disable]可是设置该事件创建后状态是否开启或关闭,默认为enable。
[comment ‘comment’]可以给该事件加上注释。
5.4、定时服务日常维护测试
5.4.1修改事件(alter event)
alter event event_name
[on schedule schedule]
[rename to new_event_name]
[on completion [not] preserve]
[comment 'comment']
[enable | disable]
[do sql_statement]
a、临时关闭事件
alter event e_test disable;
b、开启事件
alter event e_test enable;
c、将每天清空test表改为5天清空一次:
alter event e_test
on schedule every 5 day;
d、重命名事件并加上注释
alter event test.new_e_test rename to e_test comment 'e_test_cm';
5.4.2删除事件(drop event)
语法很简单,如下所示:
drop event [if exists] event_name例如删除前面创建的e_test事件
drop event e_test;当然前提是这个事件存在,否则会产生error 1513 (hy000): unknown event错误,因此最好加上if exists
drop event if exists e_test;
5.4.3查看事件
a、查看一个event的详细信息可以用下面的视图:
select * from information_schema.events where event_name = 'test_insert' and event_schema = 'test'/g;
b、简要列出所有的event:show events
语法:
show events [from schema_name]
[like 'pattern' | where expr]
格式化显示所有event
show events/g
格式化显示test用户的event
show events from test;
c、查看event的创建信息
show create event event_name
show create event test.e_test/g
5.5、结论
该特性确实非常有用,可作为定时清空数据表、监控主从服务器、汇总数据到另一张表等等,并且可以精确到每秒,实时性也可以得到保障。
不过如果当两个事件的针对相同的对象的时候,会出现冲突,这种情况还不明确是我理解的问题还是确实是这样,比如每秒插入和定时删除就会冲突。除了调度sql语句之外,mysql的调度器也可以调度存储过程。
5.6、缺点
select * from information_schema.events where event_name = 'test_insert' and event_schema = 'troaudit_db'/g;
*************************** 1. row ***************************
event_catalog: null
event_schema: troaudit_db
event_name: event_session_table
definer: egilance@%
time_zone: system
event_body: sql
event_definition: begin
call create_table_process;
end
event_type: recurring
execute_at: null
interval_value: 1800
interval_field: second
sql_mode:
starts: 2011-08-23 10:51:28
ends: null
status: enabled
on_completion: preserve
created: 2011-08-23 10:51:28
last_altered: 2011-08-23 10:51:28
last_executed: 2011-08-23 17:55:51
event_comment:
originator: 0
character_set_client: utf8
collation_connection: utf8_general_ci
database_collation: utf8_unicode_ci
1 row in set (0.00 sec)
mysql只会记录最后一次调度的时间,如果时间往前调整,小于最近执行的时间,则不会执行事件调度。
摘自:padden.zhang的专栏
bitscn.com