将分区添加到已分区 table

Adding a partition to already partitioned table

将另一个分区添加到已分区的 table 的最佳方法是什么?

原始 CREATE TABLE 语句如下所示:

CREATE TABLE `command_log` (
  `id` bigint(20) NOT NULL,
  `insert_time` datetime NOT NULL,
  `start_time` timestamp NULL DEFAULT '0000-00-00 00:00:00',
  `end_time` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00',
  `command` varchar(255) NOT NULL,
  `parameters` varchar(255) DEFAULT NULL,
  `result` mediumblob,
  `status` int(11) NOT NULL,
  PRIMARY KEY (`id`,`insert_time`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1
/*!50100 PARTITION BY RANGE (to_days(insert_time))
(PARTITION p001 VALUES LESS THAN (736237) ENGINE = InnoDB,
 PARTITION p002 VALUES LESS THAN (736268) ENGINE = InnoDB,
 PARTITION p003 VALUES LESS THAN (736298) ENGINE = InnoDB,
 ...
 PARTITION p064 VALUES LESS THAN (738156) ENGINE = InnoDB,
 PARTITION p065 VALUES LESS THAN (738187) ENGINE = InnoDB,
 PARTITION p066 VALUES LESS THAN (738215) ENGINE = InnoDB,
 PARTITION p067 VALUES LESS THAN MAXVALUE ENGINE = InnoDB)

假设我只想添加 1 个额外分区,在本例中为 p067。这是否需要完整的 ALTER TABLE 语句,例如:

ALTER TABLE command_log 
PARTITION by range (to_days(insert_time))
(
    partition p059 VALUES LESS THAN (to_days('2020-08-01'))
  , partition p060 VALUES LESS THAN (to_days('2020-09-01'))
  , partition p061 VALUES LESS THAN (to_days('2020-10-01'))
  , partition p062 VALUES LESS THAN (to_days('2020-11-01'))
  , partition p063 VALUES LESS THAN (to_days('2020-12-01'))
  , partition p064 VALUES LESS THAN (to_days('2021-01-01'))
  , partition p065 VALUES LESS THAN (to_days('2021-02-01'))
  , partition p066 VALUES LESS THAN (to_days('2021-03-01'))
  , partition p067 VALUES LESS THAN (to_days('2021-04-01'))
  , partition p068 VALUES LESS THAN (MAXVALUE)
);

如果是这样的话,究竟会发生什么?

是否删除了未包含在此声明中的旧分区(例如 p001 - p058)?

这样做是否会清除 table 中的任何现有数据(例如,p059 中的数据)?

谢谢!

您不必重新定义所有早期的分区。

在范围分区表中,将处理小于 MAXVALUE 的值的最后一个分区拆分为特定范围的几个新分区是一个常见的操作。您可以使用 REORGANIZE PARTITION 执行此操作。

例如,将最后一个分区拆分为固定日期范围的两个新分区,并在末尾添加一个新的最大值分区:

ALTER TABLE command_log REORGANIZE PARTITION p067 INTO (
  partition p067 VALUES LESS THAN (TO_DAYS('2021-04-01'))
, partition p068 VALUES LESS THAN (TO_DAYS('2021-05-01'))
, partition p069 VALUES LESS THAN (MAXVALUE)
);

此重组操作不会影响所有较早的分区。

如果在最后一个分区仍为空(即它包含零行)时执行此重组,则不需要复制数据,并且操作应该几乎是即时的。

如果您忘记进行重组,并且最后一个分区收集了 一些 行,那么重组它还为时不晚 — 但需要一点时间,与您拆分的分区中的行数成正比。重组具有数据的分区确实需要复制数据行,但只需要复制您正在重组的分区的行。较早的分区仍未触及。

有关详细信息,请参阅 https://dev.mysql.com/doc/refman/5.7/en/partitioning-management-range-list.html

我在 Mysql 8 中为最初没有任何分区的 现有 table 动态创建分区时遇到同样的问题。

我必须为未来的日期添加分区并不断添加。

我解决这个问题的步骤:

  1. 获取最新的分区名(按照包含日期的方式创建分区名)

  2. 如果分区日期不存在,则使用“范围”基本机制创建分区。 查询类似于创建分区:

    ALTER TABLE <tableName> PARTITION BY RANGE (TO_DAYS(<columnName>)) (
    PARTITION p20200527 VALUES LESS THAN (TO_DAYS('2020-05-28')),  
    PARTITION p20200528 VALUES LESS THAN (TO_DAYS('2020-05-29')),  
    PARTITION p20200529 VALUES LESS THAN (TO_DAYS('2020-05-30')),  
    PARTITION p20200530 VALUES LESS THAN MAXVALUE);
    
  3. 一旦我们在 table 上有了分区,那么我们必须使用“重组”添加分区。

    ALTER TABLE <tableName> REORGANIZE PARTITION p20200730 INTO ( 
    PARTITION p20200530 VALUES LESS THAN (TO_DAYS('2020-05-31')),  
    PARTITION p20200531 VALUES LESS THAN (TO_DAYS('2020-06-01')),  
    PARTITION p20200601 VALUES LESS THAN (TO_DAYS('2020-06-02')),  
    PARTITION p20200602 VALUES LESS THAN MAXVALUE);
    

底线是: 如果您在 table 上没有任何分区并想在任何分区上添加它日期列,您可以随时添加。一旦您使用基于“范围”的机制在 table 上创建了分区并且您想要添加更多分区,然后使用“重组”添加这些分区。