mysql表分区及批量定时建分区

我们经常在数仓hive里边使用各式分区,而在mysql表里大家可能很少使用分区。其实,mysql也有分区功能,通过分区处理能提高一些数据里大、有明显规则的增量mysql表的查询性能。下面我们来给大家介绍一些mysql的分区方法。

1.mysql分区类型

RANGE分区:基于属于一个给定连续区间的列值,把多行分配给分区。

LIST分区:类似于按RANGE分区,区别在于LIST分区是基于列值匹配一个离散值集合中的某个值来进行选择。

HASH分区:基于用户定义的表达式的返回值来进行选择的分区,该表达式使用将要插入到表中的这些行的列值进行计算。这个函数可以包含MySQL 中有效的、产生非负整数值的任何表达式。

KEY分区:类似于按HASH分区,区别在于KEY分区只支持计算一列或多列,且MySQL服务器提供其自身的哈希函数。必须有一列或多列包含整数值。

下面给大家用sql代码说明分区建立方法:

RANGE分区Sql代码:

CREATE TABLE IF NOT EXISTS `file_logs` (
  `BATCH_ID`     INT(11) NOT NULL COMMENT '任务ID',
  `FILE_SIZE`    VARCHAR(200) DEFAULT NULL COMMENT '文件大小',
  `FILE_TIME`    TIMESTAMP COMMENT '文件生成时间',
  `FILE_PATH`    VARCHAR(200) NOT NULL COMMENT '文件路径',
  `FILE_NAME`    VARCHAR(200) NOT NULL COMMENT '文件名',
  `DATA_TYPE`    VARCHAR(5) DEFAULT NULL COMMENT '数据类型',
  `DAY_ID` INT(11) NOT NULL COMMENT '开始时间',
  PRIMARY KEY (`FILE_NAME`,DAY_ID)
) ENGINE=InnoDB DEFAULT CHARSET=utf8
PARTITION BY RANGE(DAY_ID)(
PARTITION p20210101 VALUES less than (20210102),
PARTITION p20210102 VALUES less than (20210103),
PARTITION p20210103 VALUES less than (20210104));

注意:这种分区会把day_id=20210102的数据放到下一个分区p20210103 ,day_id=20210103的数据放到下一个分区p20210104…

LIST分区Sql代码:

CREATE TABLE employees (
id INT NOT NULL,
fname VARCHAR(30),
lname VARCHAR(30),
hired DATE NOT NULL DEFAULT1970-01-01,
separated DATE NOT NULL DEFAULT9999-12-31,
job_code INT,
store_id INT
)
PARTITION BY LIST(store_id)
PARTITION pNorth VALUES IN (3,5,6,9,17),
PARTITION pEast VALUES IN (1,2,10,11,19,20),
PARTITION pWest VALUES IN (4,12,13,14,18),
PARTITION pCentral VALUES IN (7,8,15,16)
);

LIST分区类似于按RANGE分区,区别在于LIST分区是基于列值匹配一个离散值集合中的某个值来进行选择

HASH分区和KEY分区比较复杂,也不常用,这里不做介绍

2. 分区操作的一些常用命令

#查看表分区

select partition_name part, partition_expression expr, partition_description descr, table_rows 
from information_schema.partitions  where table_schema = schema() and table_name='file_logs';  

#查找表分区数据

select count(*) from file_logspartition (p20210212); 

#删除分区

alter table file_logs drop partition p20210212; 

#添加分区,只能连续递增添加

alter table file_logs add partition (partition p20201227 values less than (20210228));  

#分解分区(不丢失数据)

alter table file_logs reorganize partition p20210223 into
 (partition p20210222 values less THAN (20210223),
 partition p20210223 values less than (20210224)
);

#合并分区(不丢失数据)

alter table file_logs reorganize partition p20210222,p20210223 INTO (partition p1 values less than (20210224));

3. 批量建RANG类型分区

按yyyMMdd这种日期值做分区值,每个月份天数一样,进行提前批量建分区会造成一些问题。以上面range类型分区示例,day_id是int类型,其值在月底和下个月之间数值不连续,如20210227,20210228,20210301,20210302,日期20210228和20210301并不是连续递增的。这样会在批量建分区时造成一些问题,下面我们用脚本createpartitions.sh做了一些处理,保证批量建分区日期值正常。使用crontab定时每月月初批量量建下一月的表分区。

#!/bin/bash

# mysql config
MYSQL_IP=xx.xx.xx.xx
MYSQL_PORT=xx
MYSQL_USER=xx
MYSQL_PWD=xx
MYSQL_DBS=xx
TABLE_NAME=xx

# crete partitions
day=`date -d "1 months" +"%Y%m%d"`
#day=$1
tmonth=`date -d "$day" +"%m"`
tmonthformat=`printf "%d" $tmonth`
days_list=(31 28 31 30 31 30 31 31 30 31 30 31)
day_delay=${days_list[tmonthformat-1]}

for (( i=1; i<=$day_delay; i++));do
day_id=`date -d "$i days ${day}" +"%Y%m%d"`
day_partition=`date -d "$((i+1)) days ${day}" +"%Y%m%d"`
add_partition="alter table ${TABLE_NAME} add partition(partition p${day_id} values less than (${day_partition}));"
echo "=====excute comand: $add_partition"
mysql --force -h${MYSQL_IP} -P${MYSQL_PORT} -u${MYSQL_USER} -p${MYSQL_PWD} -D${MYSQL_DBS} --local-infile=1 --default-character-set=utf8 -e "${add_partition};"
done

定时任务:

crontab -e     #或者vi /etc/crontab 编辑定时时间

* 12 1 * *  sh createpartitions.sh >> dev/null   #定时每月一号执行
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值