MySQL的分区

本文详细介绍了MySQL中的分区概念及其应用场景,包括RANGE、LIST、HASH、KEY等分区类型的特点及使用方法,以及如何进行分区管理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

MySQL的分区


什么是分区

所谓分区就是将一个表分解成多个区块进行操作和保存,从而降低每次操作的数据,提高性能。而对应用来说是透明的,从逻辑上看是只有一个表(这里跟分库分表的访问不一样),但在物理上这个表可能是由多个物理分区组成的,每个分区都是一个独立的对象,可以进行独立处理。

分区能干什么

  1. 进行逻辑数据分割,分割数据能够有多个不同的物理文件路径
  2. 可以存储更多的数据,突破系统单个文件最大限制
  3. 提升性能,提高每个分区的读写速度,提高分区范围查询的速度
  4. 可以通过删除相关分区来快速删除数据
  5. 通过跨多个磁盘来分散数据查询,从而提高磁盘I/O的性能
  6. 涉及到例如SUM()和COUNT()这样聚合函数的查询,可以很容易地进行并行处理
  7. 可以备份和恢复独立的分区,这对大数据量很有好处

分区能支持的引擎

MySQL支持大部分的存储引擎创建分区,如MyISAM、InnoDB等;不支持MERGE和CSV等来创建分区。同一个分区表中的所有分区必须是同一个存储引擎。
- 确认MySQL支持分区
从MySQL5.1开始引入分区功能,可以如下方式查看是否支持:
1. “老”的版本用:SHOW VARIABLES LIKE ‘%partition%’;
2. 新的版本用:show plugins;
- 分区类型
1. RANGE分区,基于属于一个给定连续区间的列值,把多行分配给分区
2. LIST分区,类似于按RANGE分区,LIST是列值匹配一个离散值集合中的某个值来进行选择
3. HASH分区,基于用户定义的表达式的返回值来进行选择的分区,该表达式使用将要插入到表中的这些行的列值进行计算,这个函数必须产生非负整数值
4. KEY分区,类似于按HASH分区,由MySQL服务器提供其自身的哈希函数
- 但是不论什么类型的分区,都要注意以下问题:
1. 如果表中存在primary key或者unique key时,分区的列必须是primary key或者unique key的一个组成部分,也就是说,分区函数的列只能从pk或者uk这些key中取子集
2. 如果表中不存在任何的primary key或者unique key,则可以指定任何一个列作为分区列
3. 5.5版本前的Range、List、Hash分区要求分区键必须是int;MySQL5.5及以上,支持非整型的Range和List分区,即:range columns 和 list columns。
- 分区命名
分区的名字基本上遵循其他MySQL标识符应当遵循的原则,例如用于表和数据库名字的标识符。但是应当注意,分区的名字是不区分大小写的。 无论使用何种类型的分区,分区总是在创建时就自动的顺序编号,且从0开始记录。
- 创建分区
1. RANGE分区

CREATE TABLE tbl_users (
    uuid INT NOT NULL,
    customerId VARCHAR(20), 
    pwd VARCHAR(20),
    showName VARCHAR(100), 
    trueName VARCHAR(100), 
    registerTime VARCHAR(100)
)
PARTITION BY RANGE (uuid) (
    PARTITION p0 VALUES LESS THAN (5), 
    PARTITION p1 VALUES LESS THAN (10), 
    PARTITION p2 VALUES LESS THAN (15), 
    PARTITION p3 VALUES LESS THAN MAXVALUE
);
CREATE TABLE tbl_users7( 
    uuid INT NOT NULL, 
    customerId VARCHAR(20), 
    pwd VARCHAR(20), 
    showName VARCHAR(100), 
    trueName VARCHAR(100),
    registerTime Date
)

PARTITION BY RANGE(YEAR(registerTime))

SUBPARTITION BY HASH(TO_DAYS(registerTime)) 
( 
    PARTITION p0 VALUES LESS THAN (2008)( 
        SUBPARTITION s0,

        SUBPARTITION s1 
    ),

    PARTITION p1 VALUES LESS THAN (2015)( 
        SUBPARTITION s2, 
        SUBPARTITION s3 
    ),

    PARTITION p2 VALUES LESS THAN MAXVALUE( 
        SUBPARTITION s4, 
        SUBPARTITION s5 
    ) 
);

(1)到存放数据的地方查看文件,路经配置在/usr/bin/mysql_config里面的ldata。
(2)可以通过使用形如:select * from information_schema.partitions where
table_schema=‘arch1’ and table_name=‘tbl_users’ \G; 的语句来查看分区信息
(3)可以通过形如select * from tbl_users partition(p0);的语句来查看分区上的数据
(4)可以使用形如explain partitions select * from tbl_users where uuid=2;的语句来查看MySQL会操作的分区
2. List分区

PARTITION BY List (uuid) (
    PARTITION p0 VALUES in (1,2,3,5), 
    PARTITION p1 VALUES in (7,9,10), 
    PARTITION p2 VALUES in (11,15)
); 

(1)如果试图操作的列值不在分区值列表中时,那么会失败并报错。要注意的是,LIST分区没有类似如“VALUES LESS THAN MAXVALUE”这样的包含其他值在内的定义,将要匹配的任何值都必须在值列表中找到。
(2)LIST分区除了能和RANGE分区结合起来生成一个复合的子分区,与HASH和KEY分区结合起来生成复合的子分区也是可以的。
3. Hash分区
HASH分区主要用来确保数据在预先确定数目的分区中平均分布。在RANGE和LIST分区中,必须明确指定一个给定的列值或列值集合以指定应该保存在哪个分区中;而在HASH分区中,MySQL自动完成这些工作,要做的只是基于将要被哈希的列值指定一个表达式,以及指定被分区的表将要被分割成的分区数量,如:

PARTITION BY HASH (uuid) PARTITIONS 3;

(1)由于每次插入、更新、删除一行,这个表达式都要计算一次;这意味着非常复杂的表达式可能会引起性能问题,尤其是在执行同时影响大量行的运算(例如批量插入)的时候。
(2)最有效率的哈希函数是只对单个表列进行计算,并且它的值随列值进行一致地增大或减小,因为这考虑了在分区范围上的“修剪”。也就是说,表达式值和它所基于的列的值变化越接近,就能越有效地使用该表达式来进行HASH分区。
3.1. 线性Hash分区
线性哈希分区在“PARTITION BY” 子句中添加“LINEAR”关键字。 线性哈希分区的优点在于增加、删除、合并和拆分分区将变得更加快捷,有利于处理含有极其大量数据的表。它的缺点在于各个分区间数据的分布不大可能均衡。
4. Key分区 类似于按照HASH分区,Hash分区允许用户自定义的表达式,而Key分区不允许使用用户自定义的表达式;Hash分区只支持整数分区,Key分区支持除了blob或text类型之外的其他数据类型分区。
与Hash分区不同,创建Key分区表的时候,可以不指定分区键,默认会选择使用主键或唯一键作为分区键,没有主键或唯一键,就必须指定分区键。

CREATE TABLE tbl_users4 (
    uuid INT NOT NULL,
    customerId VARCHAR(20),
    pwd VARCHAR(20),
    showName VARCHAR(100),
    trueName VARCHAR(100),
    registerTime VARCHAR(100)
)
PARTITION BY LINEAR Key (uuid) PARTITIONS 3;

f
5. 子分区
子分区是分区表中每个分区的再次分割,适合保存非常大量的数据。

CREATE TABLE tbl_users5 (
    ......
    registerTime Date 
)
PARTITION BY RANGE(YEAR(registerTime))
SUBPARTITION BY HASH(TO_DAYS(registerTime)) 
SUBPARTITIONS 2
(
    PARTITION p0 VALUES LESS THAN (2008), 
    PARTITION p1 VALUES LESS THAN (2015), 
    PARTITION p2 VALUES LESS THAN MAXVALUE
);

(1)在MySQL 5.1中,对于已经通过RANGE或LIST分区了的表再进行子分区是可能的。子分区既可以使用HASH希分区,也可以使用KEY分区。这也被称为复合分区
(2)每个分区必须有相同数量的子分区
(3)如果在一个分区表上的任何分区上使用SUBPARTITION 来明确定义任何子分区,那么就必须定义所有的子分区
(4)每个SUBPARTITION子句必须包括 (至少)子分区的一个名字
(5)在每个分区内,子分区的名字必须是唯一的,目前在整个表中,也要保持唯一。 例如:

PARTITION BY RANGE(YEAR(registerTime))
SUBPARTITION BY HASH(TO_DAYS(registerTime)) (
    PARTITION p0 VALUES LESS THAN (2008)( 
        SUBPARTITION s0,
        SUBPARTITION s1
    ),
    PARTITION p1 VALUES LESS THAN (2015)(
        SUBPARTITION s2, 
        SUBPARTITION s3 
    ),
    PARTITION p2 VALUES LESS THAN MAXVALUE( 
        SUBPARTITION s4, 
        SUBPARTITION s5
    ) 
);

子分区可以用于特别大的表,可以在多个磁盘间分配数据和索引。 例如:

SUBPARTITION s0
DATA DIRECTORY = '/disk0/data'
INDEX DIRECTORY = '/disk0/idx', SUBPARTITION s1
DATA DIRECTORY = '/disk1/data' INDEX DIRECTORY = '/disk1/idx'
  • MySQL分区处理NULL值的方式
    MySQL中的分区在禁止空值NULL上没有进行处理,无论它是一个列值还是一个用户定义表达式的值,一般而言,在这种情况下MySQL把NULL视为0。如果你希望回避这种做法,你应该在设计表时声列“NOT NULL”

分区管理概述

可以对分区进行添加、删除、重新定义、合并或拆分等管理操作。
- RANGE和LIST分区的管理
1. 删除分区语句如:alter table tbl_users drop partition p0;
(1)当删除了一个分区,也同时删除了该分区中所有的数据
(2)可以通过show create table tbl_users;来查看新的创建表的语句
(3)如果是List分区的话,删除的数据不能新增进来,因为这些行的列值包含在已经删除了的分区的值列表中
2. 添加分区语句如:alter table tbl_users add partition(partition p3 values less than(50));
(1)对于RANGE分区的表,只可以添加新的分区到分区列表的高端
(2)对于List分区的表,不能添加已经包含在现有分区值列表中的任意值
3. 如果希望能不丢失数据的条件下重新定义分区,可以使用如下语句:
ALTER TABLE tbl_name REORGANIZE PARTITION partition_list INTO
(partition_definitions)
(1)拆分分区如:

alter table tbl_users REORGANIZE PARTITION 
p1 INTO(
    partition s0 values less than(5), 
    partition s1 values less than(10));
或者如:
alter table tbl_users2 REORGANIZE PARTITION 
p0 INTO(
    partition s0 values in(1,2,3), 
    partition s1 values in(4,5));

(2)合并分区如:

alter table tbl_users2 REORGANIZE PARTITION 
s0,s1 INTO(
    partition p0 values in(1,2,3,4,5));
  1. 删除所有分区,但保留数据,形如:
    alter table tbl_users remove partitioning;
    • HASH和KEY分区的管理
      1:减少分区数量语句如:
      alter table tbl_users3 COALESCE PARTITION 2;
      2:添加分区数量语句如:
      alter table tbl_users3 add PARTITION partitions 2;
    • 其它分区管理语句
  2. 重建分区: 类似于先删除保存在分区中的所有记录,然后重新插入它们,可用于整理分区碎片。如:
    alter table tbl_users REBUILD PARTITION p2,p3;
  3. 优化分区:如果从分区中删除了大量的行,或者对一个带有可变长度的行(也就是说,有VARCHAR,BLOB或TEXT类型的列)作了许多修改,可以使用“ALTER TABLE … OPTIMIZE PARTITION”来收回没有使用的空间,并整理分区数据文件的碎片。如:
    alter table tbl_users OPTIMIZE PARTITION p2,p3;
  4. 分析分区:读取并保存分区的键分布,如:
    alter table tbl_users ANALYZE PARTITION p2,p3;
  5. 检查分区:检查分区中的数据或索引是否已经被破坏,如:
    alter table tbl_users CHECK PARTITION p2,p3;
  6. 修补分区: 修补被破坏的分区,如:
    alter table tbl_users REPAIR PARTITION p2,p3;
    • 其它
  7. 最大分区数目不能超过1024,一般建议对单表的分区数不要超过150个
  8. 如果含有唯一索引或者主键,则分区列必须包含在所有的唯一索引或者主键在内
  9. 不支持外键
  10. 不支持全文索引,对分区表的分区键创建索引,那么这个索引也将被分区
  11. 按日期进行分区很合适,因为很多日期函数可以用。但是对于字符串来说合适的分区函数不太多
  12. 只有RANG和LIST分区能进行子分区,HASH和KEY分区不能进行子分区
  13. 临时表不能被分区
  14. 分区表对于单条记录的查询没有优势
  15. 要注意选择分区的成本,每插入一行数据都需要按照表达式筛选插入的分区
  16. 分区字段尽量不要可以为null
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值