zoukankan      html  css  js  c++  java
  • Mysql

    在工业监控里面, 需要对每天的数据, 进行记录, 时间长了之后, 数据库很容易撑爆. 这时候, 如果允许, 可以对之前的数据进行一次清除, 只记录几个月内的数据.

    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中, 通过定时任务和存储过程来实现此功能, 推荐通过后台定时任务, 执行删表操作.

  • 相关阅读:
    账户与安全
    VIM 文档编辑
    ubuntu下搭建Discuz
    数据库管理及增删改查基本操作小结
    poj 3320 jessica's Reading PJroblem 尺取法 -map和set的使用
    poj 3579 Median 二分套二分 或 二分加尺取
    poj 3685 Matrix 二分套二分 经典题型
    POJ 3061  Subsequence   尺取法   挑战146页
    poj 2976 Dropping tests 二分搜索+精度处理
    Codeforces Round #325 (Div. 2) A. Alena's Schedule 暴力枚举 字符串
  • 原文地址:https://www.cnblogs.com/elvinle/p/11444781.html
Copyright © 2011-2022 走看看