MySQL中有哪些表分区类型

55次阅读
没有评论

共计 8034 个字符,预计需要花费 21 分钟才能阅读完成。

本篇文章给大家分享的是有关 MySQL 中有哪些表分区类型,丸趣 TV 小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着丸趣 TV 小编一起来看看吧。

一、什么是表分区

通俗地讲表分区是将一大表,根据条件分割成若干个小表。mysql5.1 开始支持数据表分区了。如:某用户表的记录超过了 600 万条,那么就可以根据入库日期将表分区,也可以根据所在地将表分区。当然也可根据其他的条件分区。

二、为什么要对表进行分区

为了改善大型表以及具有各种访问模式的表的可伸缩性,可管理性和提高数据库效率。

分区的一些优点包括:

与单个磁盘或文件系统分区相比,可以存储更多的数据。

对于那些已经失去保存意义的数据,通常可以通过删除与那些数据有关的分区,很容易地删除那些数据。相反地,在某些情况下,添加新数据的过程又可以通过为那些新数据专门增加一个新的分区,来很方便地实现。通常和分区有关的其他优点包括下面列出的这些。MySQL 分区中的这些功能目前还没有实现,但是在我们的优先级列表中,具有高的优先级;我们希望在 5.1 的生产版本中,能包括这些功能。

一些查询可以得到极大的优化,这主要是借助于满足一个给定 WHERE 语句的数据可以只保存在一个或多个分区内,这样在查找时就不用查找其他剩余的分区。因为分区可以在创建了分区表后进行修改,所以在第一次配置分区方案时还不曾这么做时,可以重新组织数据,来提高那些常用查询的效率。

涉及到例如 SUM()和 COUNT()这样聚合函数的查询,可以很容易地进行并行处理。这种查询的一个简单例子如“SELECT salesperson_id, COUNT (orders) as order_total FROM sales GROUP BY salesperson_id;”。通过“并行”,这意味着该查询可以在每个分区上同时进行,最终结果只需通过总计所有分区得到的结果。

通过跨多个磁盘来分散数据查询,来获得更大的查询吞吐量。

三、分区类型

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

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

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

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

RANGE 分区

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

这些区间要连续且不能相互重叠,使用 VALUES LESS THAN 操作符来进行定义。以下是实例。

Sql 代码:

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)
);

按照这种分区方案,在商店 1 到 5 工作的雇员相对应的所有行被保存在分区 P0 中,商店 6 到 10 的雇员保存在 P1 中,依次类推。注意,每个分区都是按顺序进行定义,从最低到最高。这是 PARTITION BY RANGE 语法的要求;在这点上,它类似于 C 或 Java 中的“switch … case”语句。对于包含数据 (72,‘Michael’,‘Widenius’,’1998-06-25′, NULL, 13) 的一个新行,可以很容易地确定它将插入到 p2 分区中,但是如果增加了一个编号为第 21 的商店,将会发生什么呢?在这种方案下,由于没有规则把 store_id 大于 20 的商店包含在内,服务器将不知道把该行保存在何处,将会导致错误。要避免这种错误,可以通过在 CREATE TABLE 语句中使用一个“catchall”VALUES LESS THAN 子句,该子句提供给所有大于明确指定的最高值的值:

Sql 代码:

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
);

MAXVALUE 表示最大的可能的整数值。现在,store_id 列值大于或等于 16(定义了的最高值)的所有行都将保存在分区 p3 中。在将来的某个时候,当商店数已经增长到 25, 30, 或更多,可以使用 ALTER TABLE 语句为商店 21-25, 26-30, 等等增加新的分区。在几乎一样的结构中,你还可以基于雇员的工作代码来分割表,也就是说,基于 job_code 列值的连续区间。例如——假定 2 位数字的工作代码用来表示普通(店内的)工人,三个数字代码表示办公室和支持人员,四个数字代码表示管理层,你可以使用下面的语句创建该分区表:

Sql 代码:

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 中。在 VALUES LESS THAN 子句中使用一个表达式也是可能的。这里最值得注意的限制是 MySQL 必须能够计算表达式的返回值作为 LESS THAN ()比较的一部分;因此,表达式的值不能为 NULL 。由于这个原因,雇员表的 hired, separated, job_code, 和 store_id 列已经被定义为非空(NOT NULL)。除了可以根据商店编号分割表数据外,你还可以使用一个基于两个 DATE(日期)中的一个的表达式来分割表数据。例如,假定你想基于每个雇员离开公司的年份来分割表,也就是说,YEAR(separated)的值。实现这种分区模式的 CREATE TABLE 语句的一个例子如下所示:

Sql 代码:

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
);

在这个方案中,在 1991 年前雇佣的所有雇员的记录保存在分区 p0 中,1991 年到 1995 年期间雇佣的所有雇员的记录保存在分区 p1 中,1996 年到 2000 年期间雇佣的所有雇员的记录保存在分区 p2 中,2000 年后雇佣的所有工人的信息保存在 p3 中。

RANGE 分区在如下场合特别有用:1)、当需要删除一个分区上的“旧的”数据时, 只删除分区即可。如果你使用上面最近的那个例子给出的分区方案,你只需简单地使用”ALTER TABLE employees DROP PARTITION p0;”来删除所有在 1991 年前就已经停止工作的雇员相对应的所有行。对于有大量行的表,这比运行一个如”DELETE FROM employees WHERE YEAR (separated) = 1990;”这样的一个 DELETE 查询要有效得多。2)、想要使用一个包含有日期或时间值,或包含有从一些其他级数开始增长的值的列。3)、经常运行直接依赖于用于分割表的列的查询。例如,当执行一个如”SELECT COUNT(*) FROM employees WHERE YEAR(separated) = 2000 GROUP BY store_id;”这样的查询时,MySQL 可以很迅速地确定只有分区 p2 需要扫描,这是因为余下的分区不可能包含有符合该 WHERE 子句的任何记录。

注释:这种优化还没有在 MySQL 5.1 源程序中启用,但是,有关工作正在进行中。

LIST 分区

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

LIST 分区通过使用“PARTITION BY LIST(expr)”来实现,其中“expr”是某列值或一个基于某个列值、并返回一个整数值的表达式,然后通过“VALUES IN (value_list)”的方式来定义每个分区,其中“value_list”是一个通过逗号分隔的整数列表。注释:在 MySQL 5.1 中,当使用 LIST 分区时,有可能只能匹配整数列表。

Sql 代码:

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 个有经销权的地区,如下表所示:

====================

地区   商店 ID 号

北区   3, 5, 6, 9, 17

东区   1, 2, 10, 11, 19, 20

西区   4, 12, 13, 14, 18

中心区   7, 8, 15, 16

====================

要按照属于同一个地区商店的行保存在同一个分区中的方式来分割表,可以使用下面的“CREATE TABLE”语句:

Sql 代码:

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 分区的采用上面的方案,下面的查询将失败:

Sql 代码:

INSERT INTO employees VALUES(224,  Linus ,  Torvalds ,  2002-05-01 ,  2004-10-12 , 42, 21);

这是因为“store_id”列值 21 不能在用于定义分区 pNorth, pEast, pWest, 或 pCentral 的值列表中找到。要重点注意的是,LIST 分区没有类似如“VALUES LESS THAN MAXVALUE”这样的包含其他值在内的定义。将要匹配的任何值都必须在值列表中找到。

LIST 分区除了能和 RANGE 分区结合起来生成一个复合的子分区,与 HASH 和 KEY 分区结合起来生成复合的子分区也是可能的。

HASH 分区

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

要使用 HASH 分区来分割一个表,要在 CREATE TABLE 语句上添加一个“PARTITION BY HASH (expr)”子句,其中“expr”是一个返回一个整数的表达式。它可以仅仅是字段类型为 MySQL 整型的一列的名字。此外,你很可能需要在后面再添加一个“PARTITIONS num”子句,其中 num 是一个非负的整数,它表示表将要被分割成分区的数量。

Sql 代码:

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;

如果没有包括一个 PARTITIONS 子句,那么分区的数量将默认为 1。例外:对于 NDB Cluster(簇)表,默认的分区数量将与簇数据节点的数量相同,这种修正可能是考虑任何 MAX_ROWS 设置,以便确保所有的行都能合适地插入到分区中。

LINER HASH

MySQL 还支持线性哈希功能,它与常规哈希的区别在于,线性哈希功能使用的一个线性的 2 的幂(powers-of-two)运算法则,而常规哈希使用的是求哈希函数值的模数。线性哈希分区和常规哈希分区在语法上的唯一区别在于,在“PARTITION BY”子句中添加“LINEAR”关键字。

Sql 代码:

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. 找到下一个大于 num. 的、2 的幂,我们把这个值称为 V,它可以通过下面的公式得到:2. V = POWER(2, CEILING(LOG(2, num)))(例如,假定 num 是 13。那么 LOG(2,13)就是 3.7004397181411。CEILING(3.7004397181411)就是 4,则 V = POWER(2,4), 即等于 16)。3. 设置 N = F(column_list) (V – 1). 4.  当 N = num: ·   设置 V = CEIL(V / 2) ·   设置 N = N (V – 1) 例如,假设表 t1,使用线性哈希分区且有 4 个分区,是通过下面的语句创建的:CREATE TABLE t1 (col1 INT, col2 CHAR(5), col3 DATE) PARTITION BY LINEAR HASH(YEAR(col3) ) PARTITIONS 6; 现在假设要插入两行记录到表 t1 中,其中一条记录 col3 列值为’2003-04-14′,另一条记录 col3 列值为’1998-10-19′。第一条记录将要保存到的分区确定如下:V = POWER(2, CEILING(LOG(2,7))) = 8 N = YEAR(’2003-04-14′) (8 – 1)   = 2003 7   = 3 (3 = 6 为假(FALSE): 记录将被保存到 #3 号分区中) 第二条记录将要保存到的分区序号计算如下:V = 8 N = YEAR(’1998-10-19′) (8-1)   = 1998 7   = 6 (6 = 4 为真(TRUE): 还需要附加的步骤) N = 6 CEILING(5 / 2)   = 6 3   = 2   (2 = 4 为假(FALSE): 记录将被保存到 #2 分区中) 按照线性哈希分区的优点在于增加、删除、合并和拆分分区将变得更加快捷,有利于处理含有极其大量(1000 吉)数据的表。它的缺点在于,与使用常规 HASH 分区得到的数据分布相比,各个分区间数据的分布不大可能均衡。

KSY 分区

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

Sql 代码:

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)算法得到,而不是通过模数算法。

以上就是 MySQL 中有哪些表分区类型,丸趣 TV 小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注丸趣 TV 行业资讯频道。

正文完
 
丸趣
版权声明:本站原创文章,由 丸趣 2023-08-03发表,共计8034字。
转载说明:除特殊说明外本站除技术相关以外文章皆由网络搜集发布,转载请注明出处。
评论(没有评论)