Mysql自动按月表分区

很久没有碰DB了。最近需要做一个日志表,因此重新拾了下。实现了下自动按月表分区
开发环境为Mysql 5.7.28

参考资料:
Mysql分区表及自动创建分区Partition(按日)

核心的两个存储过程:
auto_create_partition为创建表分区,调用后为该表创建到下月结束的表分区。
auto_del_partition为删除表分区,方便历史数据空间回收。

DELIMITER $$
DROP PROCEDURE IF EXISTS auto_create_partition$$
CREATE PROCEDURE `auto_create_partition`(IN `table_name` varchar(64))
BEGIN
			SET @next_month:=CONCAT(date_format(date_add(now(),interval 2 month),'%Y%m'),'01');
			SET @SQL = CONCAT( 'ALTER TABLE `', table_name, '`',
					' ADD PARTITION (PARTITION p', @next_month, " VALUES LESS THAN (TO_DAYS(",
							@next_month ,")) );" );
			PREPARE STMT FROM @SQL;
			EXECUTE STMT;
			DEALLOCATE PREPARE STMT;
END$$

DROP PROCEDURE IF EXISTS auto_del_partition$$
CREATE PROCEDURE `auto_del_partition`(IN `table_name` varchar(64),IN `reserved_month` int)
BEGIN
	DECLARE v_finished INTEGER DEFAULT 0;
	DECLARE v_part_name varchar(100) DEFAULT "";
	DECLARE part_cursor CURSOR FOR 
		select partition_name from information_schema.partitions where table_schema = schema()
			and table_name=@table_name and partition_description < TO_DAYS(CONCAT(date_format(date_sub(now(),interval reserved_month month),'%Y%m'),'01'));
	DECLARE continue handler FOR 
		NOT FOUND SET v_finished = TRUE;
	OPEN part_cursor;
read_loop: LOOP
	FETCH part_cursor INTO v_part_name;
	if v_finished = 1 then
		leave read_loop;
	end if;
	SET @SQL = CONCAT( 'ALTER TABLE `', table_name, '` DROP PARTITION ', v_part_name, ";" );
	PREPARE STMT FROM @SQL;
	EXECUTE STMT;
	DEALLOCATE PREPARE STMT;
	END LOOP;
	CLOSE part_cursor;
END$$

DELIMITER ;

下面是示例

-- 假设有个表叫records,设置分区条件为按end_time按月分区
DROP TABLE IF EXISTS `records`;
CREATE TABLE `records` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `start_time` datetime NOT NULL,
  `end_time` datetime NOT NULL,
  `memo` varchar(128) CHARACTER SET utf8mb4 NOT NULL,
  PRIMARY KEY (`id`,`end_time`)
) 
PARTITION BY RANGE (TO_DAYS(end_time))(
	PARTITION p20200801 VALUES LESS THAN ( TO_DAYS('20200801'))
);

DROP EVENT IF EXISTS `records_auto_partition`;

-- 创建一个Event,每月执行一次,同时最多保存6个月的数据
DELIMITER $$
CREATE EVENT `records_auto_partition`
ON SCHEDULE EVERY 1 MONTH ON COMPLETION PRESERVE
ENABLE
DO
BEGIN
call auto_create_partition('records');
call auto_del_partition('records',6);
END$$
DELIMITER ;

几点注意事项:

  • 对于Mysql 5.1以上版本来说,表分区的索引字段必须是主键
  • 存储过程中,DECLARE 必须紧跟着BEGIN,否则会报看不懂的错误
  • 游标的DECLARE需要在定义声明之后,否则会报错
  • 如果是自己安装的Mysql,有可能Event功能是未开启的,在创建Event时会提示错误;修改my.cnf,在 [mysqld] 下添加event_scheduler=1后重启即可。
MySQL 中,可以使用分区来提高查询性能和管理数据的效率。如果你需要按自动创建分区,可以使用 MySQL 事件调度器(Event Scheduler)和存储过程(Stored Procedure)来实现。 以下是一个示例存储过程,用于创建分区: ``` DELIMITER $$ CREATE PROCEDURE create_monthly_partitions() BEGIN DECLARE start_date DATE; DECLARE end_date DATE; SET start_date = '2022-01-01'; -- 分区开始日期 SET end_date = DATE_ADD(CURDATE(), INTERVAL 1 MONTH); -- 下一个的第一天 WHILE start_date < end_date DO SET @sql = CONCAT('ALTER TABLE my_table ADD PARTITION (PARTITION p', DATE_FORMAT(start_date, '%Y%m'), ' VALUES LESS THAN (TO_DAYS(\'', DATE_FORMAT(DATE_ADD(start_date, INTERVAL 1 MONTH), '%Y-%m-%d'), '\')))'); PREPARE stmt FROM @sql; EXECUTE stmt; DEALLOCATE PREPARE stmt; SET start_date = DATE_ADD(start_date, INTERVAL 1 MONTH); END WHILE; END$$ DELIMITER ; ``` 这个存储过程会创建从当前份开始到下一个份的所有分区,每个分区以年份和份为名称,例如 p202201、p202202、p202203 等等。 然后,你可以使用事件调度器来定期运行这个存储过程,以便自动创建新的分区。以下是一个示例事件,每的第一天凌晨 1 点运行: ``` CREATE EVENT create_monthly_partitions_event ON SCHEDULE EVERY 1 MONTH STARTS DATE_ADD(CURDATE(), INTERVAL 1 DAY) ON COMPLETION PRESERVE DO CALL create_monthly_partitions(); ``` 这个事件会在每的第一天凌晨 1 点运行,调用上面创建的存储过程来创建新的分区。 请注意,分区需要使用特定的语法来查询和管理数据,因此在使用分区时,请确保你已经理解了它们的概念和使用方法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值