MySQL分區(qū)表實(shí)現(xiàn)按月份歸類
MySQL單表數(shù)據(jù)量,建議不要超過(guò)2000W行,否則會(huì)對(duì)性能有較大影響。最近接手了一個(gè)項(xiàng)目,單表數(shù)據(jù)超7000W行,一條簡(jiǎn)單的查詢語(yǔ)句等了50多分鐘都沒(méi)出結(jié)果,實(shí)在是難受,最終,我們決定用分區(qū)表。
建表
一般的表(innodb)創(chuàng)建后只有一個(gè) idb 文件:
create table normal_table(id int primary key, no int)
查看數(shù)據(jù)庫(kù)文件:
normal_table.ibd
創(chuàng)建按月份分區(qū)的分區(qū)表,注意!除了常規(guī)主鍵外,月份字段(用來(lái)分區(qū)的字段)也必須是主鍵:
create table partition_table(id int AUTO_INCREMENT, create_date date, name varchar(10), primary key(id, create_date)) ENGINE=INNODB DEFAULT CHARSET=utf8 partition by range(month(create_date))( partition quarter1 values less than(4), partition quarter2 values less than(7), partition quarter3 values less than(10), partition quarter4 values less than(13) );
查看數(shù)據(jù)庫(kù)文件:
partition_table#p#quarter1.ibd partition_table#p#quarter2.ibd partition_table#p#quarter3.ibd partition_table#p#quarter4.ibd
插入
insert into partition_table(create_date, name) values("2021-01-25", "tom1");
insert into partition_table(create_date, name) values("2021-02-25", "tom2");
insert into partition_table(create_date, name) values("2021-03-25", "tom3");
insert into partition_table(create_date, name) values("2021-04-25", "tom4");
insert into partition_table(create_date, name) values("2021-05-25", "tom5");
insert into partition_table(create_date, name) values("2021-06-25", "tom6");
insert into partition_table(create_date, name) values("2021-07-25", "tom7");
insert into partition_table(create_date, name) values("2021-08-25", "tom8");
insert into partition_table(create_date, name) values("2021-09-25", "tom9");
insert into partition_table(create_date, name) values("2021-10-25", "tom10");
insert into partition_table(create_date, name) values("2021-11-25", "tom11");
insert into partition_table(create_date, name) values("2021-12-25", "tom12");
查詢
select count(*) from partition_table; > 12 查詢第二個(gè)分區(qū)(第二季度)的數(shù)據(jù): select * from partition_table PARTITION(quarter2); 4 2021-04-25 tom4 5 2021-05-25 tom5 6 2021-06-25 tom6
刪除
當(dāng)刪除表時(shí),該表的所有分區(qū)文件都會(huì)被刪除
補(bǔ)充:Mysql自動(dòng)按月表分區(qū)
核心的兩個(gè)存儲(chǔ)過(guò)程:
- auto_create_partition為創(chuàng)建表分區(qū),調(diào)用后為該表創(chuàng)建到下月結(jié)束的表分區(qū)。
- auto_del_partition為刪除表分區(qū),方便歷史數(shù)據(jù)空間回收。
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 ;
下面是示例
-- 假設(shè)有個(gè)表叫records,設(shè)置分區(qū)條件為按end_time按月分區(qū)
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`;
-- 創(chuàng)建一個(gè)Event,每月執(zhí)行一次,同時(shí)最多保存6個(gè)月的數(shù)據(jù)
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 ;
幾點(diǎn)注意事項(xiàng):
- 對(duì)于Mysql 5.1以上版本來(lái)說(shuō),表分區(qū)的索引字段必須是主鍵
- 存儲(chǔ)過(guò)程中,DECLARE 必須緊跟著BEGIN,否則會(huì)報(bào)看不懂的錯(cuò)誤
- 游標(biāo)的DECLARE需要在定義聲明之后,否則會(huì)報(bào)錯(cuò)
- 如果是自己安裝的Mysql,有可能Event功能是未開啟的,在創(chuàng)建Event時(shí)會(huì)提示錯(cuò)誤;修改my.cnf,在 [mysqld] 下添加event_scheduler=1后重啟即可。
到此這篇關(guān)于MySQL分區(qū)表實(shí)現(xiàn)按月份歸類的文章就介紹到這了,更多相關(guān)mysql按月表分區(qū)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
linux mysql 報(bào)錯(cuò):MYSQL:The server quit&nbs
mysql 報(bào)錯(cuò):MYSQL:The server quit without updating PID file。以下是可能的原因與解決方法2013-02-02
mysql的聯(lián)合索引(復(fù)合索引)的實(shí)現(xiàn)
這篇文章主要介紹了mysql的聯(lián)合索引(復(fù)合索引)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03
MySQL無(wú)服務(wù)及服務(wù)無(wú)法啟動(dòng)的終極解決方案分享
又是MySQL的問(wèn)題,之前已經(jīng)遇見過(guò)一次本地MySQL服務(wù)無(wú)法啟動(dòng)的情況,現(xiàn)在又出現(xiàn)了,下面這篇文章主要給大家介紹了關(guān)于MySQL無(wú)服務(wù)及服務(wù)無(wú)法啟動(dòng)的終極解決方案,需要的朋友可以參考下2022-06-06
MySQL 存儲(chǔ)引擎 MyISAM詳解(最新推薦)
使用 MyISAM 存儲(chǔ)引擎的表占用空間很小,但是由于使用表級(jí)鎖定,所以限制了讀/寫操作的性能,通常用于中小型的Web應(yīng)用和數(shù)據(jù)倉(cāng)庫(kù)配置中的只讀或主要是讀的場(chǎng)景,這篇文章主要介紹了MySQL 存儲(chǔ)引擎 MyISAM詳解,需要的朋友可以參考下2025-05-05
MySql數(shù)據(jù)庫(kù)中Select用法小結(jié)
在程序開發(fā)中數(shù)據(jù)庫(kù)是必要知識(shí)點(diǎn),今天小編給大家介紹mysql數(shù)據(jù)庫(kù)中的select用法,包括條件篩選、指定篩選和分組顯示查詢語(yǔ)句的寫法,非常不錯(cuò),對(duì)mysql select用法相關(guān)知識(shí)感興趣的朋友一起看看吧2016-10-10
DQL命令查詢數(shù)據(jù)實(shí)現(xiàn)方法詳解
DQL(Data?Query?Language,數(shù)據(jù)查詢語(yǔ)言),查詢數(shù)據(jù)庫(kù)數(shù)據(jù),如SELECT語(yǔ)句,簡(jiǎn)單的單表查詢或多表的復(fù)雜查詢和嵌套查詢,數(shù)據(jù)庫(kù)語(yǔ)言中最核心、最重要的語(yǔ)句,使用頻率最高的語(yǔ)句2022-09-09
Mysql 導(dǎo)入導(dǎo)出csv 中文亂碼問(wèn)題的解決方法
這篇文章介紹了Mysql 導(dǎo)入導(dǎo)出csv 中文亂碼問(wèn)題的解決方法,有需要的朋友可以參考一下2013-09-09
xtrabackup備份還原MySQL數(shù)據(jù)庫(kù)
這篇文章主要為大家詳細(xì)介紹了xtrabackup備份還原MySQL數(shù)據(jù)庫(kù)的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06

