RDS for MySQL 通過分區歸檔歷史數據

RDS for MySQL 通過分區歸檔歷史數據

  • 原始表

  • 分區

  • 用於分區維護的存儲過程

  • 每月調用存儲過程的事件



隨着數據的積累,數據量的增加,越來越多的表體積變的龐大,不但影響查詢的執行時間,而且使得管理工作(比如添加刪除索引)變的越發複雜和困難。

本文介紹一個通過分區滑動來歸檔歷史數據(以便可以有效控制表的尺寸)的方法,僅供大家參考。


1. 原始表

未分區的原始表:order_history。

create table order_history
( 
	id              bigint unsigned auto_increment primary key,
	order_id        bigint unsigned not null,
	customer_id     bigint unsigned not null,
	goods_id        bigint unsigned not null,
	payment_id      bigint unsigned not null,
	details         text,
	comments        text,
	order_date      datetime not null,	status          tinyint
);

 

2. 分區

MySQL 分區要求分區字段必須是主鍵或者唯一鍵的一部分,因此需要修改主鍵定義。

alter table order_history algorithm=inplace, lock=none, drop primary key, add primary key (id, order_date);-- algorithm=inplace, lock=none 是 RDS for MySQL 5.6 支持的 online DDL 特性。-- 如果要修改主鍵,刪除主鍵和添加主鍵建議放在一條語句中,以便充分利用 5.6 版本的 Online DDL 特性。alter table order_history partition by range columns (order_date) (	partition p1606 values less than ('2016-07-01'),	partition p1607 values less than ('2016-08-01'),	partition p1608 values less than ('2016-09-01'),	partition p1609 values less than ('2016-10-01'),	partition p1610 values less than ('2016-11-01'),	partition p1611 values less than ('2016-12-01'),	partition p1612 values less than ('2017-01-01'),	partition p0 values less than maxvalue
);-- 將數據按月份分區

 3. 用於分區維護的存儲過程

刪除最久的一個月份分區,新建下個月份分區,總體保留最近 6 個月份的數據。

delimiter //drop procedure sp_order_his_rotate//create procedure sp_order_his_rotate ()begin
       declare todrop_par varchar(10) default null;       declare toadd_par varchar(10) default null;       declare toadd_day varchar(10) default null;       declare last_par varchar(10) default null;       declare new_par varchar(10) default null;       declare v_sql varchar(200) default null;	   
       select date_format(date_sub(curdate(), interval 6 month), 'p%y%m') into todrop_par;       select date_format(date_add(curdate(), interval 1 month), 'p%y%m') into toadd_par;       select date_format(date_add(curdate()-day(curdate())+1,interval 2 month), '%Y-%m-%d') into toadd_day;       select partition_name into last_par from information_schema.partitions 
	 where table_name = 'order_history' 
	 and table_schema = 'db_name' 
	 and partition_description != 'maxvalue' 
	 order by partition_description asc limit 1;
       if todrop_par = last_par then       select concat('alter table order_history drop partition ',todrop_par,';') into v_sql;	 set @v_sql=v_sql;	 prepare stmt from @v_sql;	 execute stmt;	 deallocate prepare stmt;       end if;       select partition_name into new_par from information_schema.partitions 
         where table_name = 'order_history' 
         and table_schema = 'db_name' 
         and partition_description != 'maxvalue' 
	 order by partition_description desc limit 1;	
       if toadd_par != new_par then       select concat('alter table order_history reorganize partition p0 into (partition ', toadd_par, ' values less than (''', toadd_day,'''), partition p0 values less than (maxvalue));') into v_sql;	 set @v_sql=v_sql;	 prepare stmt from @v_sql;	 execute stmt;	 deallocate prepare stmt;       end if;end;
//
delimiter ;

 

4. 每月調用存儲過程的事件

每月月末 28 日調用分區管理存儲過程,來自動維護分區。

drop event if exists order_his_rotate;
delimiter //create event order_his_rotate  on schedule every 1 month starts '2016-12-28 02:00:00'
  on completion preserve enable dobegin
  call sp_order_his_rotate();end;
//

delimiter ; 



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章