5.9. 分区

PostgreSQL支持基本的表分区功能。 本节描述为什么需要表分区以及如何在数据库设计中使用表分区。

5.9.1. 概述

分区的意思是把逻辑上的一个大表分割成物理上的几块。分区可以提供若干好处:

这种好处通常只有在表可能会变得非常大的情况下才有价值。 到底多大的表会从分区中收益取决于具体的应用, 不过有个基本的拇指规则就是表的大小超过了数据库服务器的物理内存大小。

目前,PostgreSQL支持通过表继承进行分区。 每个分区必须做为单独一个父表的子表进行创建。父表自身通常是空的, 它的存在只是为了代表整个数据集。你在试图实现分区之前,应该先熟悉继承(参阅第 5.8 节)。

PostgreSQL可以实现下面形式的分区:

范围分区

表被一个或者多个关键字段分区成"范围",这些范围在不同的分区里没有重叠。 比如,我们可以通过时间范围分区,或者根据特定业务对象的标识符范围分区。

列表分区

表通过明确地列出每个分区里应该出现哪些关键字值实现。

5.9.2. 实现分区

要设置一个分区的表,做下面的步骤:

  1. 创建"主表",所有分区都从它继承。

    这个表中没有数据,不要在这个表上定义任何检查约束,除非你希望约束同样也适用于所有分区。 同样,在其上定义任何索引或者唯一约束也没有意义。

  2. 创建几个"子表",每个都从主表上继承。通常,这些表不会增加任何字段。

    我们将把子表称作分区,尽管它们就是普通的PostgreSQL表。

  3. 给分区表增加约束,定义每个分区允许的健值。

    典型的例子是:

    CHECK ( x = 1 )
    CHECK ( county IN ( 'Oxfordshire', 'Buckinghamshire', 'Warwickshire' ))
    CHECK ( outletID >= 100 AND outletID < 200 )

    确保这些约束能够保证在不同的分区里不会有重叠的键值。一个常见的错误是设置下面这样的范围:

    CHECK ( outletID BETWEEN 100 AND 200 )
    CHECK ( outletID BETWEEN 200 AND 300 )

    这样做是错误的,因为它没说清楚健值 200 属于那个范围。

    请注意在范围和列表分区的语法方面没有什么区别;这些术语只是用于描述的。

  4. 对于每个分区,在关键字字段上创建一个索引,以及其它你想创建的索引。 关键字字段索引并非严格必需的,但是在大多数情况下它是很有帮助的。 如果你希望关键字值是唯一的,那么你应该总是给每个分区创建一个唯一或者主键约束。

  5. 另外,定义一个规则或者触发器,来重定向数据插入主表到适当的分区。

  6. 确保postgresql.conf里的配置参数constraint_exclusion是打开的。 没有这个参数,查询不会按照需要进行优化。

比如,假设我们为一个巨大的冰激凌公司构造数据库。该公司每天都测量最高温度, 以及每个地区的冰激凌销售。概念上,我们需要一个这样的表:

CREATE TABLE measurement (
    city_id         int not null,
    logdate         date not null,
    peaktemp        int,
    unitsales       int
);

我们知道大多数查询都只会访问最后一周,最后一个月或者最后一个季度的数据, 因为这个表的主要用途是为管理准备在线报告。为了减少需要存储的旧数据, 我们决定值保留最近三年的有用数据。在每个月的开头,我们都会删除最旧的一个月的数据。

在这种情况下,我们可以使用分区来帮助实现所有对表的不同需求。 下面的步骤描述了上面的需求,分区可以这样设置:

  1. 主表是measurement表,就像上面那样声明。

  2. 然后我们为每个月创建一个分区:

    CREATE TABLE measurement_y2006m02 ( ) INHERITS (measurement);
    CREATE TABLE measurement_y2006m03 ( ) INHERITS (measurement);
    ...
    CREATE TABLE measurement_y2007m11 ( ) INHERITS (measurement);
    CREATE TABLE measurement_y2007m12 ( ) INHERITS (measurement);
    CREATE TABLE measurement_y2008m01 ( ) INHERITS (measurement);

    每个分区都是拥有自己内容的完整的表,只是它们从measurement表继承定义。

    这样就解决了我们的一个问题:删除旧数据。每个月, 我们需要做的只是在最旧的子表上执行一个DROP TABLE, 然后为新月份创建一个新的子表。

  3. 我们必须提供非重叠的表约束。而不是只像上面那样创建分区表,所以我们的建表脚本就变成:

    CREATE TABLE measurement_y2006m02 (
        CHECK ( logdate >= DATE '2006-02-01' AND logdate < DATE '2006-03-01' )
    ) INHERITS (measurement);
    CREATE TABLE measurement_y2006m03 (
        CHECK ( logdate >= DATE '2006-03-01' AND logdate < DATE '2006-04-01' )
    ) INHERITS (measurement);
    ...
    CREATE TABLE measurement_y2007m11 (
        CHECK ( logdate >= DATE '2007-11-01' AND logdate < DATE '2007-12-01' )
    ) INHERITS (measurement);
    CREATE TABLE measurement_y2007m12 (
        CHECK ( logdate >= DATE '2007-12-01' AND logdate < DATE '2008-01-01' )
    ) INHERITS (measurement);
    CREATE TABLE measurement_y2008m01 (
        CHECK ( logdate >= DATE '2008-01-01' AND logdate < DATE '2008-02-01' )
    ) INHERITS (measurement);

  4. 我们可能还需要在关键字字段上有索引:

    CREATE INDEX measurement_y2006m02_logdate ON measurement_y2006m02 (logdate);
    CREATE INDEX measurement_y2006m03_logdate ON measurement_y2006m03 (logdate);
    ...
    CREATE INDEX measurement_y2007m11_logdate ON measurement_y2007m11 (logdate);
    CREATE INDEX measurement_y2007m12_logdate ON measurement_y2007m12 (logdate);
    CREATE INDEX measurement_y2008m01_logdate ON measurement_y2008m01 (logdate);

    我们选择先不建立更多的索引。

  5. 我们想让我们的应用可以说INSERT INTO measurement ... 并且数据被重定向到相应的分区表。我们可以安排给主表附上一个合适的触发器。 如果数据只进入最新的分区,我们可以使用一个非常简单的触发器:

    CREATE OR REPLACE FUNCTION measurement_insert_trigger()
    RETURNS TRIGGER AS $$
    BEGIN
        INSERT INTO measurement_y2008m01 VALUES (NEW.*);
        RETURN NULL;
    END;
    $$
    LANGUAGE plpgsql;

    创建完函数后,我们将创建一个调用触发器函数的触发器:

    CREATE TRIGGER insert_measurement_trigger
        BEFORE INSERT ON measurement
        FOR EACH ROW EXECUTE PROCEDURE measurement_insert_trigger();

    我们必须每月重新定义触发器,以便它总是指向当前分区。然而,触发定义不需要更新。

    我们可能想插入数据并且想让服务器自动定位应该向哪个分区插入数据。 我们可以用下面这个复杂的触发器来实现这个目标,比如:

    CREATE OR REPLACE FUNCTION measurement_insert_trigger()
    RETURNS TRIGGER AS $$
    BEGIN
        IF ( NEW.logdate >= DATE '2006-02-01' AND
             NEW.logdate < DATE '2006-03-01' ) THEN
            INSERT INTO measurement_y2006m02 VALUES (NEW.*);
        ELSIF ( NEW.logdate >= DATE '2006-03-01' AND
                NEW.logdate < DATE '2006-04-01' ) THEN
            INSERT INTO measurement_y2006m03 VALUES (NEW.*);
        ...
        ELSIF ( NEW.logdate >= DATE '2008-01-01' AND
                NEW.logdate < DATE '2008-02-01' ) THEN
            INSERT INTO measurement_y2008m01 VALUES (NEW.*);
        ELSE
            RAISE EXCEPTION 'Date out of range.  Fix the measurement_insert_trigger() function!';
        END IF;
        RETURN NULL;
    END;
    $$
    LANGUAGE plpgsql;

    每一个触发器跟以前一样。注意,每一个IF测试必须匹配其分区的 CHECK约束。

    当这个函数比单月的情况更复杂时,它不需要经常的更新,因为分支可以在需要之前被添加。

    注意: 在实践中,如果大部分插入该分区,它可能最好首先检查最新分区。 为简单起见,我们已经在这个例子中的其他部分表明在同一顺序下的触发器的测试。

我们可以看出,一个复杂的分区方案可能要求相当多的 DDL 。 在上面的例子里我们需要每个月创建一次新分区, 因此写一个脚本自动生成需要的 DDL 是明智的。

5.9.3. 管理分区

通常分区集在定义表的时候就已经确定了,但我们常常需要周期性的删除旧分区并添加新分区。 分区最重要的好处是它能恰到好处的适应这个需求:以极快的速度操作分区的结构, 而不是痛苦的物理移动大量数据。

删除旧数据最简单的方法是删除不再需要的分区:

DROP TABLE measurement_y2006m02;

这个命令可以迅速删除数包含数百万条记录的分区,因为它不需要单独删除每一条记录。

还可以在删除分区的同时保留其作为一个表访问的能力:

ALTER TABLE measurement_y2006m02 NO INHERIT measurement;

这将允许将来对这些数据执行其它的操作(比如使用COPY,pg_dump 之类的工具进行备份)。并且此时也是执行其它数据操作(数据聚合或运行报表等)的有利时机。

同样,我们可以像前面创建最初的分区一样,创建一个新的空分区来处理新数据。

CREATE TABLE measurement_y2008m02 (
    CHECK ( logdate >= DATE '2008-02-01' AND logdate < DATE '2008-03-01' )
) INHERITS (measurement);

有时在分区结构之外创建新表并在一段时间之后将其变为分区更为方便。 因为这将允许在该表变为分区之前对其中的数据进行加载、检查、转换之类的操作。

CREATE TABLE measurement_y2008m02
  (LIKE measurement INCLUDING DEFAULTS INCLUDING CONSTRAINTS);
ALTER TABLE measurement_y2008m02 ADD CONSTRAINT y2008m02
   CHECK ( logdate >= DATE '2008-02-01' AND logdate < DATE '2008-03-01' );
\copy measurement_y2008m02 from 'measurement_y2008m02'
--其它可能的数据准备工作
ALTER TABLE measurement_y2008m02 INHERIT measurement;

5.9.4. 分区和约束排除

约束排除是一种查询优化技巧,它改进了用上述方法定义的表分区的性能。比如:

SET constraint_exclusion = on;
SELECT count(*) FROM measurement WHERE logdate >= DATE '2008-01-01';

如果没有约束排除,上面的查询会扫描measurement表中的每一个分区。 打开了约束排除之后,规划器将检查每个分区的约束然后试图证明该分区不需要被扫描 (因为它不能包含任何符合WHERE子句条件的数据行)。如果规划器可以证明这个, 它就把该分区从查询规划里排除出去。

你可以使用EXPLAIN命令显示一个规划在constraint_exclusion 打开和关闭情况下的不同。一个为这种类型的表设置的典型的非最佳的规划是:

SET constraint_exclusion = off;
EXPLAIN SELECT count(*) FROM measurement WHERE logdate >= DATE '2008-01-01';

                                          QUERY PLAN
-----------------------------------------------------------------------------------------------
 Aggregate  (cost=158.66..158.68 rows=1 width=0)
   ->  Append  (cost=0.00..151.88 rows=2715 width=0)
         ->  Seq Scan on measurement  (cost=0.00..30.38 rows=543 width=0)
               Filter: (logdate >= '2008-01-01'::date)
         ->  Seq Scan on measurement_y2006m02 measurement  (cost=0.00..30.38 rows=543 width=0)
               Filter: (logdate >= '2008-01-01'::date)
         ->  Seq Scan on measurement_y2006m03 measurement  (cost=0.00..30.38 rows=543 width=0)
               Filter: (logdate >= '2008-01-01'::date)
...
         ->  Seq Scan on measurement_y2007m12 measurement  (cost=0.00..30.38 rows=543 width=0)
               Filter: (logdate >= '2008-01-01'::date)
         ->  Seq Scan on measurement_y2008m01 measurement  (cost=0.00..30.38 rows=543 width=0)
               Filter: (logdate >= '2008-01-01'::date)

部分或者全部分区可能会使用索引扫描而不是全表扫描, 不过这里要表达的意思是没有必要扫描旧分区就可以回答这个查询。 在打开约束排除之后,我们可以得到生成同样回答的明显简化的规划:

SET constraint_exclusion = on;
EXPLAIN SELECT count(*) FROM measurement WHERE logdate >= DATE '2008-01-01';
                                          QUERY PLAN
-----------------------------------------------------------------------------------------------
 Aggregate  (cost=63.47..63.48 rows=1 width=0)
   ->  Append  (cost=0.00..60.75 rows=1086 width=0)
         ->  Seq Scan on measurement  (cost=0.00..30.38 rows=543 width=0)
               Filter: (logdate >= '2008-01-01'::date)
         ->  Seq Scan on measurement_y2008m01 measurement  (cost=0.00..30.38 rows=543 width=0)
               Filter: (logdate >= '2008-01-01'::date)

请注意,约束排除只由CHECK约束驱动,而不会由索引驱动。 因此,在关键字字段上定义索引是没有必要的。 在给出的分区上是否需要建立索引取决于那些扫描该分区的查询通常是扫描该分区的一大部分还是只是一小部分。 对于后者,索引通常都有帮助,对于前者则没有什么好处。

constraint_exclusion缺省(和建议)设置事实上不是on 也不是off,但是中间设置调用partition, 导致很可能要工作在分区表上的技术只适用于查询。on 设置导致规划器在所有的查询里检查CHECK限制,即使是不可能受益的最简单的限制。

5.9.5. 替代分区方法

用一个不同的途径去重新定向插入适当的分区表是在主表中建立规则,而不是触发器,例如:

CREATE RULE measurement_insert_y2006m02 AS
ON INSERT TO measurement WHERE
    ( logdate >= DATE '2006-02-01' AND logdate < DATE '2006-03-01' )
DO INSTEAD
    INSERT INTO measurement_y2006m02 VALUES (NEW.*);
...
CREATE RULE measurement_insert_y2008m01 AS
ON INSERT TO measurement WHERE
    ( logdate >= DATE '2008-01-01' AND logdate < DATE '2008-02-01' )
DO INSTEAD
    INSERT INTO measurement_y2008m01 VALUES (NEW.*);

规则比触发器有显著的开销,但是这个开销是每检查一次支付一次而不是每行支付一次, 所以这种方法可能在批量插入的情况下有优势。然而在更多的情况下, 触发器的方法更好。

请注意COPY会忽略规则。如果您想用COPY插入数据, 您将需要复制分区表而不是主表。COPY触发触发器, 如果您用触发器的方法就可以正常使用。

另一个规则方法缺点是如果规则设置没有覆盖插入数据,那么没有简单的路径强制错误, 数据将会悄悄代替主表中的数据。

安排分区也可以用UNION ALL视图,而不是表继承。例如,

CREATE VIEW measurement AS
          SELECT * FROM measurement_y2006m02
UNION ALL SELECT * FROM measurement_y2006m03
...
UNION ALL SELECT * FROM measurement_y2007m11
UNION ALL SELECT * FROM measurement_y2007m12
UNION ALL SELECT * FROM measurement_y2008m01;

然而,增加和删除各个分区的数据集,需要重新创建视图,增加一个额外的步骤。 在实际中这个方法跟使用继承相比较几乎没有可取之处。

5.9.6. 警告

下面的注意事项适合于已分区的表:

下面的注意事项适合于约束排除: