一、什么是表分区
通俗地讲表分区是将一大表,根据条件分割成若干个小表。mysql5.1开始支持数据表分区了。
如:某用户表的记录超过了600万条,那么就可以根据入库日期将表分区,也可以根据所在地将表分区。当然也可根据其他的条件分区。
二、为什么要对表进行分区
为了改善大型表以及具有各种访问模式的表的可伸缩性,可管理性和提高数据库效率。
分区的一些优点包括:
三、分区类型
·
RANGE分区:基于属于一个给定连续区间的列值,把多行分配给分区。
· LIST分区:类似于按RANGE分区,区别在于LIST分区是基于列值匹配一个离散值集合中的某个值来进行选择。
·
HASH分区:基于用户定义的表达式的返回值来进行选择的分区,该表达式使用将要插入到表中的这些行的列值进行计算。这个函数可以包含MySQL
中有效的、产生非负整数值的任何表达式。
· KEY分区:类似于按HASH分区,区别在于KEY分区只支持计算一列或多列,且MySQL
服务器提供其自身的哈希函数。必须有一列或多列包含整数值。
- RANGE分区
- CREATE
TABLE employees ( -
id INT NOT NULL, -
fname VARCHAR(30), -
lname VARCHAR(30), -
hired DATE NOT NULL DEFAULT '1970-01-01', -
separated DATE NOT NULL DEFAULT '9999-12-31', -
job_code INT NOT NULL, -
store_id INT NOT NULL - )
-
- partition
BY RANGE (store_id) ( -
partition p0 VALUES LESS THAN (6), -
partition p1 VALUES LESS THAN (11), -
partition p2 VALUES LESS THAN (16), -
partition p3 VALUES LESS THAN (21) -
);
- CREATE
TABLE employees ( -
id INT NOT NULL, -
fname VARCHAR(30), -
lname VARCHAR(30), -
hired DATE NOT NULL DEFAULT '1970-01-01', -
separated DATE NOT NULL DEFAULT '9999-12-31', -
job_code INT NOT NULL, -
store_id INT NOT NULL - )
-
- PARTITION
BY RANGE (store_id) ( -
PARTITION p0 VALUES LESS THAN (6), -
PARTITION p1 VALUES LESS THAN (11), -
PARTITION p2 VALUES LESS THAN (16), -
PARTITION p3 VALUES LESS THAN MAXVALUE - );
- CREATE
TABLE employees ( -
id INT NOT NULL, -
fname VARCHAR(30), -
lname VARCHAR(30), -
hired DATE NOT NULL DEFAULT '1970-01-01', -
separated DATE NOT NULL DEFAULT '9999-12-31', -
job_code INT NOT NULL, -
store_id INT NOT NULL - )
-
- PARTITION
BY RANGE (job_code) ( -
PARTITION p0 VALUES LESS THAN (100), -
PARTITION p1 VALUES LESS THAN (1000), -
PARTITION p2 VALUES LESS THAN (10000) - );
在这个例子中,
店内工人相关的所有行将保存在分区p0中,办公室和支持人员相关的所有行保存在分区p1中,管理层相关的所有行保存在分区p2中。
- CREATE
TABLE employees ( -
id INT NOT NULL, -
fname VARCHAR(30), -
lname VARCHAR(30), -
hired DATE NOT NULL DEFAULT '1970-01-01', -
separated DATE NOT NULL DEFAULT '9999-12-31', -
job_code INT, -
store_id INT - )
-
- PARTITION
BY RANGE (YEAR(separated)) ( -
PARTITION p0 VALUES LESS THAN (1991), -
PARTITION p1 VALUES LESS THAN (1996), -
PARTITION p2 VALUES LESS THAN (2001), -
PARTITION p3 VALUES LESS THAN MAXVALUE - );
RANGE分区在如下场合特别有用:
注释:这种优化还没有在MySQL 5.1源程序中启用,但是,有关工作正在进行中。
- LIST分区
注释:在MySQL 5.1中,当使用LIST分区时,有可能只能匹配整数列表。
- CREATE
TABLE employees ( -
id INT NOT NULL, -
fname VARCHAR(30), -
lname VARCHAR(30), -
hired DATE NOT NULL DEFAULT '1970-01-01', -
separated DATE NOT NULL DEFAULT '9999-12-31', -
job_code INT, -
store_id INT - );
假定有20个音像店,分布在4个有经销权的地区,如下表所示:
====================
地区
------------------------------------
北区
东区
西区
中心区
====================
要按照属于同一个地区商店的行保存在同一个分区中的方式来分割表,可以使用下面的“CREATE TABLE”语句:
- CREATE
TABLE employees ( -
id INT NOT NULL, -
fname VARCHAR(30), -
lname VARCHAR(30), -
hired DATE NOT NULL DEFAULT '1970-01-01', -
separated DATE NOT NULL DEFAULT '9999-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) - );
这使得在表中增加或删除指定地区的雇员记录变得容易起来。例如,假定西区的所有音像店都卖给了其他公司。那么与在西区音像店工作雇员相关的所有记录
(行)可以使用查询“ALTER TABLE employees DROP PARTITION
pWest;”来进行删除,它与具有同样作用的DELETE (删除)查询“DELETE query DELETE FROM
employees WHERE store_id IN (4,12,13,14,18);”比起来,要有效得多。
【要点】:如果试图插入列值(或分区表达式的返回值)不在分区值列表中的一行时,那么“INSERT”查询将失败并报错。例如,假定LIST分区的采用上面的方案,下面的查询将失败:
这是因为“store_id”列值21不能在用于定义分区pNorth, pEast,
pWest,或pCentral的值列表中找到。要重点注意的是,LIST分区没有类似如“VALUES
LESS THAN
MAXVALUE”这样的包含其他值在内的定义。将要匹配的任何值都必须在值列表中找到。
LIST分区除了能和RANGE分区结合起来生成一个复合的子分区,与HASH和KEY分区结合起来生成复合的子分区也是可能的。
- HASH分区
- CREATE
TABLE employees ( -
id INT NOT NULL, -
fname VARCHAR(30), -
lname VARCHAR(30), -
hired DATE NOT NULL DEFAULT '1970-01-01', -
separated DATE NOT NULL DEFAULT '9999-12-31', -
job_code INT, -
store_id INT - )
- PARTITION
BY HASH(store_id) -
PARTITIONS
4;
这种修正可能是考虑任何MAX_ROWS 设置,以便确保所有的行都能合适地插入到分区中。
- LINER HASH
MySQL还支持线性哈希功能,它与常规哈希的区别在于,线性哈希功能使用的一个线性的2的幂(powers-of-two)运算法则,而常规
哈希使用的是求哈希函数值的模数。
线性哈希分区和常规哈希分区在语法上的唯一区别在于,在“PARTITION BY” 子句中添加“LINEAR”关键字。
- CREATE
TABLE employees ( -
id INT NOT NULL, -
fname VARCHAR(30), -
lname VARCHAR(30), -
hired DATE NOT NULL DEFAULT '1970-01-01', -
separated DATE NOT NULL DEFAULT '9999-12-31', -
job_code INT, -
store_id INT - )
- PARTITION
BY LINEAR HASH(YEAR(hired)) -
PARTITIONS
4;
假设一个表达式expr, 当使用线性哈希功能时,记录将要保存到的分区是num
个分区中的分区N,其中N是根据下面的算法得到:
1.
2.
(例如,假定num是13。那么LOG(2,13)就是3.7004397181411。
CEILING(3.7004397181411)就是4,则V = POWER(2,4), 即等于16)。
3.
4.
·
·
例如,假设表t1,使用线性哈希分区且有4个分区,是通过下面的语句创建的:
CREATE TABLE t1 (col1 INT, col2 CHAR(5), col3 DATE)
现在假设要插入两行记录到表t1中,其中一条记录col3列值为'2003-04-14',另一条记录col3列值为'1998-10-19'。第一条记录将要保存到的分区确定如下:
V = POWER(2, CEILING(LOG(2,7))) = 8
N = YEAR('2003-04-14') & (8 - 1)
(3 >= 6 为假(FALSE): 记录将被保存到#3号分区中)
第二条记录将要保存到的分区序号计算如下:
V = 8
N = YEAR('1998-10-19') & (8-1)
(6 >= 4 为真(TRUE): 还需要附加的步骤)
N = 6 & CEILING(5 / 2)
(2 >= 4 为假(FALSE): 记录将被保存到#2分区中)
按照线性哈希分区的优点在于增加、删除、合并和拆分分区将变得更加快捷,有利于处理含有极其大量(1000吉)数据的表。它的缺点在于,与使用
常规HASH分区得到的数据分布相比,各个分区间数据的分布不大可能均衡。
- KSY分区
类似于按HASH分区,区别在于KEY分区只支持计算一列或多列,且MySQL 服务器提供其自身的哈希函数。必须有一列或多列包含整数值。
- CREATE
TABLE tk ( -
col1 INT NOT NULL, -
col2 CHAR(5), -
col3 DATE - )
- PARTITION
BY LINEAR KEY (col1) -
PARTITIONS
3;
在KEY分区中使用关键字LINEAR和在HASH分区中使用具有同样的作用,分区的编号是通过2的幂(powers-of-two)算法得到,而不是通过模数算法。