在工业监控里面, 需要对每天的数据, 进行记录, 时间长了之后, 数据库很容易撑爆. 这时候, 如果允许, 可以对之前的数据进行一次清除, 只记录几个月内的数据.
delimiter $ DROP PROCEDURE if exists p_clearOldData; /* g_date_limit 时间限制, 如 2019_08 , 则删除 该时间之前的表 g_date_length 时间格式的长度, 如 后缀时间是 2019_08, 则此处应该传 7 */ CREATE PROCEDURE `p_clearOldData`(in g_date_limit varchar(30) , in g_date_length int ) begin /*查询到的表名*/ DECLARE g_table VARCHAR(100); /*查询到的表名对应的后缀时间*/ DECLARE g_date VARCHAR(30) DEFAULT ''; /*定义done, 用于跳出循环使用*/ DECLARE done bit DEFAULT 0; ###DECLARE g_date_limit VARCHAR(30) DEFAULT ''; ###DECLARE g_date_length int default 10; /*声明游标*/ DECLARE g_cursor CURSOR FOR select TABLE_NAME from INFORMATION_SCHEMA.TABLES where TABLE_SCHEMA='abc' and TABLE_NAME like 'tbl_abc_%'; /*游标查询时, 如果找不到下一个了, 会将done置为1, 用于跳出 REPEAT 循环*/ DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = 1; /*待执行动态sql*/ set @v_full_sql := ''; ###set g_date_limit = DATE_FORMAT(date_add(NOW(), interval -90 day),'%Y_%m_%d'); /*打开游标*/ open g_cursor; /*循环*/ REPEAT FETCH g_cursor into g_table; set g_date = right(g_table, g_date_length); if g_date < g_date_limit then set @v_full_sql = CONCAT('drop table if exists ',g_table); /*预编译此动态sql, 并存入stmt中*/ PREPARE stmt from @v_full_sql; /*执行此动态sql, 此动态sql的作用, 是删除表*/ execute stmt; /*释放此资源*/ DEALLOCATE PREPARE stmt; /*对执行情况进行记录*/ INSERT into tbl_event_log (name, time) values (g_table, NOW()); end if; /*结束repeat循环*/ UNTIL done END REPEAT; /*关闭游标*/ close g_cursor; select 'OK'; end $ delimiter;
在数据库中, 进行测试:
CREATE TABLE `t_bk001_2019_02` ( `id` int(11) NOT NULL AUTO_INCREMENT, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8
执行脚本:
call p_clearOldData('2019_03', 7);
会发现, 表确实被删除了, 且别的表并未受到影响.
在不能发后台包的情况下, 可以通过mysql定时任务和存储过程, 来实现定时删表操作.
不过, 如果通过这种方式, 还需要对此存储过程进行改动, 或者再创建一个存储过程, 对此进行封装成一个没有参数的存储过程.
对于这种定时任务的执行情况和执行时机, 最好能加入一个日志进去:
CREATE TABLE `tbl_event_log` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(100) DEFAULT NULL COMMENT '表名', `time` datetime DEFAULT NULL COMMENT '时间', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8
*注:
以上操作, 不推荐在mysql中, 通过定时任务和存储过程来实现此功能, 推荐通过后台定时任务, 执行删表操作.