当前位置:Gxlcms > 数据库问题 > MySQL 分区表探究细节《整理自官档》

MySQL 分区表探究细节《整理自官档》

时间:2021-07-01 10:21:17 帮助过:13人阅读

1.   Date Time分区

When partitioning by KEY or LINEAR KEY, youcan use a DATE, TIME, or DATETIME column as the

partitioning column without performing any modification of the column value. Forexample, this table creation statement is perfectly valid in MySQL:

如果考虑不转换列值,可采取key进行分区。

CREATE TABLE members(

firstname VARCHAR(25)NOT NULL,

lastname VARCHAR(25)NOT NULL,

username VARCHAR(16)NOT NULL,

email VARCHAR(35),

joined DATE NOT NULL

)

PARTITION BYKEY(joined)

PARTITIONS 6;

这样分区,不用转换列值

 

In MySQL 5.7, it is also possible to use aDATE or DATETIME column as the partitioning column using RANGE COLUMNS and LISTCOLUMNS partitioning. MySQL‘s other partitioning types, however, require apartitioning expression that yields an integer value or NULL. If you wish touse date-based partitioning by RANGE, LIST, HASH, or LINEAR HASH, you can simplyemploy a function that operates on a DATE, TIME, or DATETIME column and returnssuch a value, as shown here:

如果想使用基于日期RANGE, LIST, HASH, or LINEAR HASH的分区,可使用时间函数进行分区。

CREATE TABLE members(

firstname VARCHAR(25)NOT NULL,

lastname VARCHAR(25)NOT NULL,

username VARCHAR(16)NOT NULL,

email VARCHAR(35),

joined DATE NOT NULL

)

PARTITION BY RANGE(YEAR(joined) ) (

PARTITION p0 VALUESLESS THAN (1960),

PARTITION p1 VALUESLESS THAN (1970),

PARTITION p2 VALUESLESS THAN (1980),

PARTITION p3 VALUESLESS THAN (1990),

PARTITION p4 VALUESLESS THAN MAXVALUE

);

MySQL partitioning is optimized for usewith the TO_DAYS(), YEAR(), and TO_SECONDS()

functions. However, you can use other dateand time functions that return an integer or NULL, such

as WEEKDAY(), DAYOFYEAR(), or MONTH().

可以使用时间日期函数(返回整数或null)来进行分区优化,如TO_DAYS(), YEAR(), and TO_SECONDS(), WEEKDAY(), DAYOFYEAR(), orMONTH()

PS:分区表分区名,大小写不敏感。

mysql> CREATE TABLE t2 (val INT)

   ->  PARTITION BY LIST(val)(

   ->  PARTITION mypart VALUES IN(1,3,5),

   ->  PARTITION MyPart VALUES IN(2,4,6)

   ->  );

ERROR 1517 (HY000): Duplicate partitionname MyPart

 

2.   Range分区

案例1——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 (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

);

案例2——range date

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

);

案例3——range timestamp

CREATE TABLE quarterly_report_status (

report_id INT NOT NULL,

report_status VARCHAR(20) NOT NULL,

report_updated TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMPON UPDATE CURRENT_TIMESTAMP

)

PARTITION BY RANGE ( UNIX_TIMESTAMP(report_updated) ) (

PARTITION p0 VALUES LESS THAN (UNIX_TIMESTAMP(‘2008-01-01 00:00:00‘) ),

PARTITION p1 VALUES LESS THAN (UNIX_TIMESTAMP(‘2008-04-01 00:00:00‘) ),

PARTITION p2 VALUES LESS THAN (UNIX_TIMESTAMP(‘2008-07-01 00:00:00‘) ),

PARTITION p3 VALUES LESS THAN (UNIX_TIMESTAMP(‘2008-10-01 00:00:00‘) ),

PARTITION p4 VALUES LESS THAN (UNIX_TIMESTAMP(‘2009-01-01 00:00:00‘) ),

PARTITION p5 VALUES LESS THAN ( UNIX_TIMESTAMP(‘2009-04-0100:00:00‘) ),

PARTITION p6 VALUES LESS THAN (UNIX_TIMESTAMP(‘2009-07-01 00:00:00‘) ),

PARTITION p7 VALUES LESS THAN (UNIX_TIMESTAMP(‘2009-10-01 00:00:00‘) ),

PARTITION p8 VALUES LESS THAN (UNIX_TIMESTAMP(‘2010-01-01 00:00:00‘) ),

PARTITION p9 VALUES LESS THAN (MAXVALUE)

);

案列4——range columns(date or datetime):

此部分案例可以参考文档MySQL分区——Column Partition.docx

CREATE TABLE members (

firstname VARCHAR(25) NOT NULL,

lastname VARCHAR(25) NOT NULL,

username VARCHAR(16) NOT NULL,

email VARCHAR(35),

joined datetime NOTNULL

)

PARTITION BY RANGE COLUMNS(joined) (

PARTITION p0 VALUES LESS THAN (‘1960-01-0100:00:00‘),

PARTITION p1 VALUES LESS THAN (‘1970-01-0100:00:00‘),

PARTITION p2 VALUES LESS THAN (‘1980-01-0100:00:00‘),

PARTITION p3 VALUES LESS THAN (‘1990-01-0100:00:00‘),

PARTITION p4 VALUES LESS THAN MAXVALUE

);

CREATE TABLE members (

firstname VARCHAR(25) NOT NULL,

lastname VARCHAR(25) NOT NULL,

username VARCHAR(16) NOT NULL,

email VARCHAR(35),

joined date NOTNULL

)

PARTITION BY RANGE COLUMNS(joined) (

PARTITION p0 VALUES LESS THAN (‘1960-01-0100:00:00‘),

PARTITION p1 VALUES LESS THAN (‘1970-01-0100:00:00‘),

PARTITION p2 VALUES LESS THAN (‘1980-01-0100:00:00‘),

PARTITION p3 VALUES LESS THAN (‘1990-01-0100:00:00‘),

PARTITION p4 VALUES LESS THAN MAXVALUE

);

3.   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

)

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)

);

4.   Columns Partitioning

The permitted data types are shown in the followinglist:支持的数据类型如下

(整数)All integer types: TINYINT, SMALLINT,MEDIUMINT, INT (INTEGER), and BIGINT. (This is the

same as with partitioning by RANGE andLIST.)

Other numeric data types (such as DECIMALor FLOAT) are not supported as partitioning columns.

(日期)DATE and DATETIME.

Columns using other data types relating todates or times are not supported as partitioning columns.

(字符串)The following string types: CHAR, VARCHAR,BINARY, and VARBINARY.

COLUMNS Partitioning

TEXT and BLOB columns are not supported aspartitioning columns.

  1. 1.    

  2. 2.    

  3. 3.    

  4. 4.    

  5. 1.    

  6. 2.    

  7. 3.    

  8. 4.    

  9. 1.   Range Columns

实验

create table rcx

 (aint,

  bint,

  cchar(10),

  dint)

partition by range columns(a, b, c)

 (partition p0 values less than (5, 10, ‘aaa‘),

 partition p1 values less than (10, 20, ‘nnnn‘),

 partition pmax values less than (maxvalue, maxvalue, maxvalue));

 

insert into rcx values  (5, 9, ‘aaa‘, 1);   -- insert into p0

insert into rcx values  (5, 10, ‘aaa‘, 1);  -- insert into p1

insert into rcx values  (5, 11, ‘aaa‘, 1);  -- insert into p1

insert into rcx values  (4, 12, ‘aaa‘, 1)   -- insert into p0;

select (5, 9) < (5, 10), (5, 10) <(5, 10), (4, 12) < (5, 10) ;

# (5, 9) < (5, 10), (5, 10) < (5,10), (4, 12) < (5, 10)

1, 0, 1

备注:是按照数组进行比较插入的,官方文档参考位置21.2.3.1RANGE COLUMNS partitioning

以下SQL都是能正确创建的,来自官档。

案例1

CREATE TABLE rc2 (

a INT,

b INT

)

PARTITION BY RANGE COLUMNS(a,b) (

PARTITION p0 VALUES LESS THAN (0,10),

PARTITION p1 VALUES LESS THAN (10,20),

PARTITION p2 VALUES LESS THAN (10,30),

PARTITION p3 VALUES LESS THAN(MAXVALUE,MAXVALUE)

);

案例2

CREATE TABLE rc3 (

a INT,

b INT

)

PARTITION BY RANGE COLUMNS(a,b) (

PARTITION p0 VALUES LESS THAN (0,10),

PARTITION p1 VALUES LESS THAN (10,20),

PARTITION p2 VALUES LESS THAN (10,30),

PARTITION p3 VALUES LESS THAN (10,35),

PARTITION p4 VALUES LESS THAN (20,40),

PARTITION p5 VALUES LESS THAN(MAXVALUE,MAXVALUE)

);

案例3

CREATE TABLE rc4 (

a INT,

b INT,

c INT

)

PARTITION BY RANGE COLUMNS(a,b,c) (

PARTITION p0 VALUES LESS THAN (0,25,50),

PARTITION p1 VALUES LESS THAN (10,20,100),

PARTITION p2 VALUES LESS THAN (10,30,50)

PARTITION p3 VALUES LESS THAN(MAXVALUE,MAXVALUE,MAXVALUE)

);

When designing tablespartitioned by RANGE COLUMNS, you can always test successive partition

definitions by comparingthe desired tuples using the mysql client, like this:

当你设计range columns分区表,通过比较期望的数组来测试连续分区定义

 

mysql> SELECT (0,25,50) <(10,20,100), (10,20,100) < (10,30,50);

+-------------------------+--------------------------+

| (0,25,50) <(10,20,100) | (10,20,100) < (10,30,50) |

+-------------------------+--------------------------+

| 1 | 1 |

+-------------------------+--------------------------+

1 row in set (0.00 sec)

案例4

CREATE TABLE employees_by_lname (

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 COLUMNS (lname) (

PARTITION p0 VALUES LESS THAN (‘g‘),

PARTITION p1 VALUES LESS THAN (‘m‘),

PARTITION p2 VALUES LESS THAN (‘t‘),

PARTITION p3 VALUES LESS THAN (MAXVALUE)

);

 

普通表转为rangecolumns

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

);

ALTER TABLE employees PARTITION BY RANGECOLUMNS (hired) (

PARTITION p0 VALUES LESS THAN(‘1970-01-01‘),

PARTITION p1 VALUES LESS THAN(‘1980-01-01‘),

PARTITION p2 VALUES LESS THAN(‘1990-01-01‘),

PARTITION p3 VALUES LESS THAN(‘2000-01-01‘),

PARTITION p4 VALUES LESS THAN(‘2010-01-01‘),

PARTITION p5 VALUES LESS THAN (MAXVALUE)

);

ALTER TABLE employees PARTITION BY RANGECOLUMNS (lname) (

PARTITION p0 VALUES LESS THAN (‘g‘),

PARTITION p1 VALUES LESS THAN (‘m‘),

PARTITION p2 VALUES LESS THAN (‘t‘),

PARTITION p3 VALUES LESS THAN (MAXVALUE)

);

注意:listrange分区不支持非int的分区list columnsrangecolumns分区支持条件见第4大点的首部。

  1. 4.1.    

  2. 2.   List Columns

案例1:

CREATE TABLE customers_1 (

first_name VARCHAR(25),

last_name VARCHAR(25),

street_1 VARCHAR(30),

street_2 VARCHAR(30),

city VARCHAR(15),

renewal DATE

)

PARTITION BY LIST COLUMNS(city) (

PARTITION pRegion_1 VALUES IN(‘Oskarshamn‘,‘Hgsby‘, ‘Mnsters‘),

PARTITION pRegion_2 VALUES IN(‘Vimmerby‘,‘Hultsfred‘, ‘Vstervik‘),

PARTITION pRegion_3 VALUES IN(‘Nssj‘,‘Eksj‘, ‘Vetlanda‘),

PARTITION pRegion_4 VALUES IN(‘Uppvidinge‘,‘Alvesta‘, ‘Vxjo‘)

);

As with partitioning by RANGE COLUMNS, youdo not need to use expressions in the COLUMNS() clause to convert column valuesinto integers. (In fact, the use of expressions other than column names is not permittedwith COLUMNS().)

实际上和range columns一样,你不需要将columns中的表达式转换为integer(实际上,不允许在columns中使用除了列名之外的表达式)。

就是表达式不允许使用。

案例2

CREATE TABLE customers_2 (

first_name VARCHAR(25),

last_name VARCHAR(25),

street_1 VARCHAR(30),

street_2 VARCHAR(30),

city VARCHAR(15),

renewal DATE

)

PARTITION BY LIST COLUMNS(renewal) (

PARTITION pWeek_1 VALUES IN(‘2010-02-01‘,‘2010-02-02‘, ‘2010-02-03‘,

‘2010-02-04‘, ‘2010-02-05‘, ‘2010-02-06‘,‘2010-02-07‘),

PARTITION pWeek_2 VALUES IN(‘2010-02-08‘,‘2010-02-09‘, ‘2010-02-10‘,

‘2010-02-11‘, ‘2010-02-12‘, ‘2010-02-13‘,‘2010-02-14‘),

PARTITION pWeek_3 VALUES IN(‘2010-02-15‘,‘2010-02-16‘, ‘2010-02-17‘,

‘2010-02-18‘, ‘2010-02-19‘, ‘2010-02-20‘,‘2010-02-21‘),

PARTITION pWeek_4 VALUES IN(‘2010-02-22‘,‘2010-02-23‘, ‘2010-02-24‘,

‘2010-02-25‘, ‘2010-02-26‘, ‘2010-02-27‘,‘2010-02-28‘)

);

案例3——含复合列

create table test_list_cols

 (col1 char(4),

 col2 char(4))

 partition by list columns(col1, col2)

 (partition part_1 values in (‘a‘, ‘b‘),

   partition part_2 values in (‘c‘, ‘d‘));

这个SQL是创建失败的,syntax错误。

create table test_list_cols

 (col1 char(4),

 col2 char(4))

 partition by list columns(col1, col2)

 (partition part_1 values in ((‘a‘, ‘b‘)),

  partition part_2 values in ((‘c‘, ‘d‘)));

和单列list不一样,需多个括号(tuple)。

5.   Hash partition

To partition a table using HASHpartitioning, it is necessary to append to the CREATE TABLE statement aPARTITION BY HASH (expr) clause, where expr is anexpression that returns an integer. This can simply be the name of acolumn whose type is one of MySQL‘s integer types. In addition, you most likelywant to follow this with PARTITIONS num, where num is a positive integerrepresenting the number of partitions into which the table is to be divided.

即分区键必须是integer type, 或返回integer type的表达式

案例1

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;

案例2——返回整数expr

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( YEAR(hired) )

PARTITIONS 4;

expr must return a nonconstant,nonrandom integer value (in other words, it should be varying

but deterministic), You should also keep in mind that this expression is evaluatedeach time a row is inserted or updated (or possibly deleted); this means thatvery complex expressions may give rise to performance issues, particularly whenperforming operations (such as batch inserts) that affect a great many rows atone time.

expression表达式返回的值,应该是非常量非随机的整数,即数值上不同且确定性的。表达式的复杂程度,可能造成批量插入的性能问题。

6.   Key Partition

Partitioning by key is similar topartitioning by hash, except that where hash partitioning employs a user-definedexpression, the hashing function for key partitioningis supplied by the MySQL server.

MySQL Cluster uses MD5()for this purpose; for tables using other storage engines, the server

employs its own internal hashing functionwhich is based on the same algorithm as PASSWORD().

The syntax rules for CREATE TABLE ...PARTITION BY KEY are similar to those for creating a

table that is partitioned by hash. Themajor differences are listed here:

KEY is used rather than HASH.

KEY takes only a list of zero or morecolumn names. Any columns used as the partitioning key

must comprise part or all of the table‘sprimary key, if the table has one. Where no column name is specified as thepartitioning key, the table‘s primary key is used, if there is one.

Key partition分区的hash函数是由mysql服务器提供,涉及到不同的算法。

Key后面可包含0或多个列,如果表中有主键,Key()中没指定列,则主键被使用为key

案列1

CREATE TABLE k1 (

id INT NOT NULL PRIMARY KEY,

name VARCHAR(20)

)

PARTITION BY KEY()

PARTITIONS 2;

If there is no primary key but there is aunique key, then the unique key is used for the partitioning

key

如果没有主键但有唯一键,唯一键(必须非空)被作为分区键。

案列2

CREATE TABLE k1 (

id INT NOT NULL,

name VARCHAR(20),

UNIQUE KEY (id)

)

PARTITION BY KEY()

PARTITIONS 2;

Unlike the case with other partitioningtypes, columns used for partitioning by KEY are not restricted to integer orNULL values.

不像其他分区类型一样,KEY分区的分区键不限于整数或null

案列3——CHAR

CREATE TABLE tm1 (

s1 CHAR(32) PRIMARY KEY

)

PARTITION BY KEY(s1)

PARTITIONS 10;

案列4——linear key:

CREATE TABLE tk (

col1 INT NOT NULL,

col2 CHAR(5),

col3 DATE

)

PARTITION BY LINEARKEY (col1)

PARTITIONS 3;

7.   Subpartition

案列1

CREATE TABLE ts (id INT, purchased DATE)

PARTITION BY RANGE( YEAR(purchased) )

SUBPARTITION BY HASH( TO_DAYS(purchased) )

SUBPARTITIONS 2 (

PARTITION p0 VALUES LESS THAN (1990),

PARTITION p1 VALUES LESS THAN (2000),

PARTITION p2 VALUES LESS THAN MAXVALUE

);

In MySQL 5.7, it is possible tosubpartition tables that are partitioned by RANGE or LIST. Subpartitions mayuse either HASH or KEY partitioning.

分区使用rangelist,子分区使用hashkey。注意subpartition by key必须明确指定列

案列2

CREATE TABLE ts (id INT, purchased DATE)

PARTITION BY RANGE( YEAR(purchased) )

SUBPARTITION BY HASH( TO_DAYS(purchased) )(

PARTITION p0 VALUES LESS THAN (1990) (

SUBPARTITION s0,

SUBPARTITION s1

),

PARTITION p1 VALUES LESS THAN (2000) (

SUBPARTITION s2,

SUBPARTITION s3

),

PARTITION p2 VALUES LESS THAN MAXVALUE (

SUBPARTITION s4,

SUBPARTITION s5

)

);

注意:

  1. 每个分区有相同数量的子分区;

  2. 如果显式指定分区和子分区,需全部指定完整;

  3. 子分区名字必须唯一;

案列3——myiasm指定不同的存储

CREATE TABLE ts (id INT, purchased DATE)

ENGINE = MYISAM

PARTITION BY RANGE( YEAR(purchased) )

SUBPARTITION BY HASH( TO_DAYS(purchased) )(

PARTITION p0 VALUES LESS THAN (1990) (

SUBPARTITION s0

DATA DIRECTORY = ‘/disk0/data‘

INDEX DIRECTORY = ‘/disk0/idx‘,

SUBPARTITION s1

DATA DIRECTORY = ‘/disk1/data‘

INDEX DIRECTORY = ‘/disk1/idx‘

),

PARTITION p1 VALUES LESS THAN (2000) (

SUBPARTITION s2

DATA DIRECTORY = ‘/disk2/data‘

INDEX DIRECTORY = ‘/disk2/idx‘,

SUBPARTITION s3

DATA DIRECTORY = ‘/disk3/data‘

INDEX DIRECTORY = ‘/disk3/idx‘

),

PARTITION p2 VALUES LESS THAN MAXVALUE (

SUBPARTITION s4

DATA DIRECTORY = ‘/disk4/data‘

INDEX DIRECTORY = ‘/disk4/idx‘,

SUBPARTITION s5

DATA DIRECTORY = ‘/disk5/data‘

INDEX DIRECTORY = ‘/disk5/idx‘

)

);

 

CREATE TABLE ts (id INT, purchased DATE)

ENGINE = MYISAM

PARTITION BY RANGE(YEAR(purchased))

SUBPARTITION BY HASH( TO_DAYS(purchased) )(

PARTITION p0 VALUES LESS THAN (1990) (

SUBPARTITION s0a

DATA DIRECTORY = ‘/disk0‘

INDEX DIRECTORY = ‘/disk1‘,

SUBPARTITION s0b

DATA DIRECTORY = ‘/disk2‘

INDEX DIRECTORY = ‘/disk3‘

),

PARTITION p1 VALUES LESS THAN (2000) (

SUBPARTITION s1a

DATA DIRECTORY = ‘/disk4/data‘

INDEX DIRECTORY = ‘/disk4/idx‘,

SUBPARTITION s1b

DATA DIRECTORY = ‘/disk5/data‘

INDEX DIRECTORY = ‘/disk5/idx‘

),

PARTITION p2 VALUES LESS THAN MAXVALUE (

SUBPARTITION s2a,

SUBPARTITION s2b

)

);

8.   Handle Null

Range Partition直接插入最小的分区内;

List Partition只有显式定义nullvalue,才能插入空值,如partition part_1 values in (null)

Hash & Key Partition中,Null被当做0来处理。

9.   Partition Management

In MySQL 5.7, all partitions of apartitioned table must have the same number of

subpartitions, and it is not possible tochange the subpartitioning once the table

has been created.

mysql5.7中,分区表的所有分区必须拥有相同数量的子分区;一旦分区表创建,不能改变子分区。

CREATE TABLE trb3 (id INT, nameVARCHAR(50), purchased DATE)

PARTITION BY RANGE( YEAR(purchased) ) (

PARTITION p0 VALUES LESS THAN (1990),

PARTITION p1 VALUES LESS THAN (1995),

PARTITION p2 VALUES LESS THAN (2000),

PARTITION p3 VALUES LESS THAN (2005)

);

insert into trb3 values(1, ‘test‘,‘1988-10-10‘), (2, ‘test‘, ‘1999-10-10‘), (3, ‘test‘, ‘2001-10-10‘);

ALTER TABLE trb3 PARTITION BY KEY(id)PARTITIONS 2;

这里可以成功修改的,即使trb3里有数据,即分区表有数据,也可重定义表结构。但是在oracle里,是不允许这样修改的。

  1. 5.    

  2. 6.    

  3. 7.    

  4. 8.    

  5. 9.    

  6. 1.   Range & List

删除分区

alter table droppartition xxx;这个语法和oracle一致。

If you wish to drop all data from allpartitions while preserving the table definition and its partitioning scheme,use the TRUNCATE TABLE statement.

如果想删除数据同时保留表结构,truncate table xxx

If you intend to change the partitioning ofa table without losing data, use ALTER TABLE ...

REORGANIZE PARTITION instead.

如果想改变表分区不丢失数据,使用alter table t reorganize partition xxx;

添加分区

Range添加分区:alter table members add partition (partition p3values less than (2000));

这个语法和oracle不太一样,oracle的语法如下:

alter table members add partition p3 valuesless than (2000);

List添加分区:ALTER TABLE tt ADD PARTITION (PARTITION p2 VALUESIN (7, 14, 21));

同时添加多个分区

ALTER TABLE employees ADD PARTITION (

PARTITION p5 VALUES LESS THAN (2010),

PARTITION p6 VALUES LESS THAN MAXVALUE

);

分区合并与拆分

Range

alter table members reorganize partition p0into (

 partition s0 values less than (1960),

 partition s1 values less than (1970));

alter table members reorganize partitions0, s1 into (

 partition p0 values less than (1970));

ALTER TABLE members REORGANIZE PARTITIONp0,p1,p2,p3 INTO (

PARTITION m0 VALUES LESS THAN (1980),

PARTITION m1 VALUES LESS THAN (2000)

);

List

ALTER TABLE tt ADD PARTITION (PARTITION npVALUES IN (4, 8));

ALTER TABLE tt REORGANIZE PARTITION p1,npINTO (

PARTITION p1 VALUES IN (6, 18),

PARTITION np VALUES in (4, 8, 12)

);

注意事项:

You cannot use REORGANIZE PARTITION tochange the table‘s partitioning type; that is, you cannot

(for example) change RANGE partitions toHASH partitions or vice versa. You also cannot use this

command to change the partitioningexpression or column. To accomplish either of these tasks

without dropping and re-creating the table,you can use ALTER TABLE ... PARTITION BY ....

不能使用reorganize来改变分区类型,也不能用于改变分区表达式或列。为了达到上述目的且不删除数据和重构表,可以使用ALTER TABLE ... PARTITION BY ....

9.2.   Hash & Key

这里包括linear hashlinear key

减少分区

alter table clientscoalesce partition 4;

增加分区:

alter table clients addpartition partitions 2;

增减都是指增减的分区个数。

9.3.   Exchange data

分区数据

CREATE TABLE e (

id INT NOT NULL,

fname VARCHAR(30),

lname VARCHAR(30)

)

PARTITION BY RANGE (id) (

PARTITION p0 VALUES LESS THAN (50),

PARTITION p1 VALUES LESS THAN (100),

PARTITION p2 VALUES LESS THAN (150),

PARTITION p3 VALUES LESS THAN (MAXVALUE)

);

INSERT INTO e VALUES

(1669, "Jim", "Smith"),

(337, "Mary", "Jones"),

(16, "Frank", "White"),

(2005, "Linda","Black");

create table e2 like e;

alter table e2 removepartitioning;

alter table e exchangepartition p0 with table e2;

e2是普通表。

mysql> ALTER TABLE e EXCHANGE PARTITIONp0 WITH TABLE e2;

ERROR 1707 (HY000): Found row that does notmatch the partition

The IGNORE keyword is accepted, but has noeffect when used with EXCHANGE PARTITION, as

shown here:

mysql> ALTER IGNORE TABLE e EXCHANGEPARTITION p0 WITH TABLE e2;

ERROR 1707 (HY000): Found row that does notmatch the partition

Only the WITHOUT VALIDATION option wouldpermit this operation to succeed:

mysql> ALTERTABLE e EXCHANGE PARTITION p0 WITH TABLE e2 WITHOUT VALIDATION;

Query OK, 0 rows affected (0.02 sec)

交换分区数据,不需确认。

with validation, 这个不常用。

子分区数据:

ALTER TABLE es EXCHANGEPARTITION p3sp0 WITH TABLE es2;

p3sp0是子分区,含子分区的表不支持分区的交换

mysql> ALTER TABLE es EXCHANGE PARTITIONp3 WITH TABLE es2;

ERROR 1704 (HY000): Subpartitioned table,use subpartition instead of partition

还需保持需交换的分区和普通表的存储引擎相同。

9.4.   Maintenance of Partitions

分区表的维护。

分区的重构:

alter table t rebuildpartition p0, p1;

作用分区重构,整理分区碎片,删除分区数据,重新插入数据。

分区的优化:

ALTER TABLE t1 OPTIMIZEPARTITION p0, p1;

当删除大量数据或修改分区中变长字段(如varchar,blob,text字段),回收未使用空间和重新整理碎片。

等效于check partition, analyze partition, repair patition

Some MySQL storage engines, including InnoDB, do not support per-partition optimization; in

these cases, ALTERTABLE ... OPTIMIZE PARTITION analyzes and rebuilds the entire table,

and causes an appropriate warning to beissued. (Bug #11751825, Bug #42822) Use ALTER

TABLE ... REBUILD PARTITION and ALTER TABLE... ANALYZE PARTITION instead, to

avoid this issue.

分区的分析:

ALTER TABLE t1 ANALYZEPARTITION p3;

This reads and stores the key distributionsfor partitions.

读取分区的分布和存储。

分区的修复:

ALTER TABLE t1 REPAIRPARTITION p0,p1;

修复崩溃的分区。

Normally, REPAIRPARTITION fails when the partition contains duplicate key errors. In MySQL

5.7.2 and later, you can use ALTER IGNORETABLE with this option, in which case all rows that

cannot be moved due to the presence ofduplicate keys are removed from the partition (Bug

#16900947).

分区的检查:

ALTER TABLE trb3 CHECKPARTITION p1;

作用和check TABLE with nonpartitioned tables一样。

会检查分区的数据和索引是否正常,用ALTER TABLE ... REPAIR PARTITION修复。

Normally, CHECKPARTITION fails when the partition contains duplicate key errors. In MySQL 5.7.2

and later, you can use ALTER IGNORE TABLEwith this option, in which case the statement returns

the contents of each row in the partitionwhere a duplicate key violation is found. Note that only the values for thecolumns in the partitioning expression for the table are reported. (Bug#16900947)

注意

The use of mysqlcheckand myisamchk is not supported withpartitioned tables.

mysqlcheckmyisamchk不支持分区表。

In MySQL 5.7, you can also truncatepartitions using ALTER TABLE ... TRUNCATE PARTITION.

This statement can be used to delete allrows from one or more partitions in much the same way that TRUNCATE TABLEdeletes all rows from a table.

可以使用ALTER TABLE ... TRUNCATE PARTITION.

ALTER TABLE ... TRUNCATEPARTITION ALL truncates all partitions in the table.

Prior to MySQL 5.7.2, ANALYZE, CHECK,OPTIMIZE, REBUILD, REPAIR, and TRUNCATE operations

were not permitted on subpartitions (Bug#14028340, Bug #65184).

5.7.2版本之前,不支持子分区的操作(ANALYZE, CHECK, OPTIMIZE, REBUILD, REPAIR, and TRUNCATE

9.5.   Obtaining Information AboutPartitions

获取分区信息。

Using the SHOW CREATE TABLEstatement to view the partitioning clauses used in creating a

partitioned table.

Using the SHOW TABLE STATUSstatement to determine whether a table is partitioned.

Querying the INFORMATION_SCHEMA.PARTITIONStable.

Using the statement EXPLAIN SELECTto see which partitions are used by a given SELECT.

mysql> show create table t \G;

mysql> show table status like ‘t‘ \G;

EXPLAIN SELECT * FROM t WHERE id < 5 \G

10.Partitionspruning

意思,就是mysql optimizer根据条件,能准确定位数据在哪个(或哪些)分区,达到性能提升的目的。亲,可以理解为mysql服务器自己的查询优化。

注意:

When pruning is performed on a partitionedMyISAM table, all partitions are opened, whether or not they are examined, dueto the design of the MyISAM storage engine.

当使用MyISAM分区表时,所有分区都将被打开,无论分区是否被检查。

MySQL can apply partition pruning toSELECT, DELETE, and UPDATE statements. INSERT statements

currently cannot be pruned.

Mysql优化器能将分区裁剪应用于SELECT, DELETE, and UPDATE语句中,但是insert语句不支持version5.7.17

补充:

Pruning can also be applied for tablespartitioned on a DATE or DATETIME column whenthe partitioning expression uses the YEAR() or TO_DAYS()function.In addition, in MySQL 5.7, pruning can beapplied for such tables when the partitioning expression uses the TO_SECONDS()function.

案例1——range

CREATE TABLE t2 (

fname VARCHAR(50) NOT NULL,

lname VARCHAR(50) NOT NULL,

region_code TINYINT UNSIGNED NOT NULL,

dob DATE NOT NULL

)

PARTITION BY RANGE( YEAR(dob) ) (

PARTITION d0 VALUES LESS THAN (1970),

PARTITION d1 VALUES LESS THAN (1975),

PARTITION d2 VALUES LESS THAN (1980),

PARTITION d3 VALUES LESS THAN (1985),

PARTITION d4 VALUES LESS THAN (1990),

PARTITION d5 VALUES LESS THAN (2000),

PARTITION d6 VALUES LESS THAN (2005),

PARTITION d7 VALUES LESS THAN MAXVALUE

);

SELECT * FROM t2 WHERE dob = ‘1982-06-23‘;

UPDATE t2 SET region_code = 8 WHERE dobBETWEEN ‘1991-02-15‘ AND ‘1997-04-25‘;

DELETE FROM t2 WHERE dob >= ‘1984-06-21‘AND dob <= ‘1999-06-21‘;

 

Invalid DATE and DATETIME values referencedin the WHERE condition of a statement against a partitioned table are treatedas NULL.

where子句中无效的DATEDATETIME将被视为NULL,不返回任何值。

案例2——list

CREATE TABLE t3 (

fname VARCHAR(50) NOT NULL,

lname VARCHAR(50) NOT NULL,

region_code TINYINT UNSIGNED NOT NULL,

dob DATE NOT NULL

)

PARTITION BY LIST(region_code) (

PARTITION r0 VALUES IN (1, 3),

PARTITION r1 VALUES IN (2, 5, 8),

PARTITION r2 VALUES IN (4, 9),

PARTITION r3 VALUES IN (6, 7, 10)

);

查询

SELECT * FROM t3 WHERE region_code BETWEEN1 AND 3

案例3——key

CREATE TABLE t4 (

fname VARCHAR(50) NOT NULL,

lname VARCHAR(50) NOT NULL,

region_code TINYINT UNSIGNED NOT NULL,

dob DATE NOT NULL

)

PARTITION BY KEY(region_code)

PARTITIONS 8;

 

UPDATE t4 WHERE region_code = 7;

SELECT * FROM t4 WHERE region_code > 2AND region_code < 6;

SELECT * FROM t4 WHERE region_code BETWEEN3 AND 5;

11.PartitionsSelection

显式分区选择:

SQL statements supporting explicit partition selection are listed here:

SELECT

DELETE

INSERT

REPLACE

UPDATE

LOAD DATA.

LOAD XML.

SELECT * FROM employeesPARTITION (p1, p2),中间以逗号分隔,这种写法在oracle里不支持的。

When a table is created using [LINEAR] HASHor [LINEAR] KEY partitioning and the names of the partitions are not specified,MySQL automatically names the partitions p0, p1, p2, ..., pN-1, where N is thenumber of partitions. For subpartitions not explicitly named, MySQL assignsautomatically to the subpartitions in each partition pX the names pXsp0, pXsp1,pXsp2, ..., pXspM-1, where M is the number of subpartitions.

当表以[LINEAR] HASH or [LINEAR] KEY分区时并且分区名字未被指定,MySQL自动命名分区p0, p1,p2, ..., pN-1。对于没有显式定义的子分区,MySQL自动命名子分区pXsp0, pXsp1, pXsp2, ..., pXspM-1

案例1

mysql> CREATE TABLE employees_sub (

-> id INT NOT NULL AUTO_INCREMENT,

-> fname VARCHAR(25) NOT NULL,

-> lname VARCHAR(25) NOT NULL,

-> store_id INT NOT NULL,

-> department_id INT NOT NULL,

-> PRIMARY KEY pk (id, lname)

-> )

-> PARTITION BY RANGE(id)

-> SUBPARTITION BY KEY (lname)

-> SUBPARTITIONS 2 (

-> PARTITION p0 VALUES LESS THAN (5),

-> PARTITION p1 VALUES LESS THAN (10),

-> PARTITION p2 VALUES LESS THAN (15),

-> PARTITION p3 VALUES LESS THANMAXVALUE

-> );

Query OK, 0 rows affected (1.14 sec)

mysql> SELECT id, CONCAT(fname, ‘ ‘,lname) AS name

-> FROM employees_sub PARTITION (p2sp1);  ——仅是这里换成了子分区名字

案例2——在join中使用显式分区

mysql> SELECT

-> e.id AS ‘Employee ID‘,CONCAT(e.fname, ‘ ‘, e.lname) AS Name,

-> s.city AS City, d.name AS department

Partition Selection

3436

-> FROM employees AS e

-> JOIN storesPARTITION (p1) AS s ON e.store_id=s.id

-> JOIN departmentsPARTITION (p0) AS d ON e.department_id=d.id

-> ORDER BY e.lname;

案例3——多个分区插入

For statements that write multiple rows toa partitioned table that uses the InnoDB storage engine:

If any row in the list following VALUEScannot be written to one of the partitions specified in the

partition_names list, the entire statementfails and no rows are written.

对应多行记录同时插入以InnoDB为引擎的分区表,如果某条记录和显式指定的分区名不匹配,则整个语句都将失败,不会写入任何行。

mysql> SHOW CREATE TABLE employees\G

*************************** 1. row***************************

Table: employees

Create Table: CREATE TABLE `employees` (

`id` int(11) NOT NULL AUTO_INCREMENT,

`fname` varchar(25) NOT NULL,

`lname` varchar(25) NOT NULL,

`store_id` int(11) NOT NULL,

`department_id` int(11) NOT NULL,

PRIMARY KEY (`id`)

) ENGINE=InnoDB AUTO_INCREMENT=27 DEFAULTCHARSET=latin1

/*!50100 PARTITION BY RANGE (id)

(PARTITION p0 VALUES LESS THAN (5) ENGINE =InnoDB,

PARTITION p1 VALUES LESS THAN (10) ENGINE =InnoDB,

PARTITION p2 VALUES LESS THAN (15) ENGINE =InnoDB,

PARTITION p3 VALUES LESS THAN (20) ENGINE =InnoDB,

PARTITION p4 VALUES LESS THAN (25) ENGINE =InnoDB,

PARTITION p5 VALUES LESS THAN MAXVALUEENGINE = InnoDB) */

1 row in set (0.00 sec)

mysql> INSERT INTO employees PARTITION(p3, p4) VALUES

-> (24, ‘Tim‘, ‘Greene‘, 3, 1), (26,‘Linda‘, ‘Mills‘, 2, 1);

ERROR 1729 (HY000): Found a row not matching the given partition set

mysql> INSERT INTO employees PARTITION(p3, p4. p5) VALUES

-> (24, ‘Tim‘, ‘Greene‘, 3, 1), (26,‘Linda‘, ‘Mills‘, 2, 1);

Query OK, 2 rows affected (0.06 sec)

Records: 2 Duplicates: 0 Warnings: 0

12.分区表的限制

  1. 分区表达式中禁用的结构:

Stored procedures,stored functions, UDFs, or plugins.

Declared variables or user variables.

  1. 算术和逻辑运算符:

Use of thearithmetic operators +, -, and * is permitted in

partitioning expressions. However, the result must be aninteger value or NULL

  1. Handler statements

Previously, theHANDLER statement was not supported with partitioned

tables. This limitation is removed beginning with MySQL5.7.1.

  1. Server SQL mode

仅说最关键的一点:SQL_MODE请在数据库建立之初就指定并且不要发生变化

  1. 性能考虑:

File system operations.

Partitioning and repartitioning operations(such as ALTER TABLE with PARTITION BY ..., REORGANIZE PARTITION, or REMOVEPARTITIONING) depend on

file systemoperations for their implementation. This means thatthe speed of these operations is affected by such factors as

人气教程排行