关于统一多级分区表

如果一个多级分区(MLP)表是一个统一分区表,GPORCA支持在该MLP上的查询。多级分区表是用SUBPARTITION子句创建的分区表。统一分区表必须满足下面这些条件。

  • 分区表结构统一。同一层上的每一个分区节点必须具有相同的层次结构。
  • 分区键约束必须一致且统一。在每一个子分区层次上,为每个分支创建的子表上的约束集必须匹配。

可以用几种方式显示分区表的信息,包括显示来自这些来源的信息:

  • pg_partitions系统视图包含分区表结构的信息。
  • pg_constraint系统catalog包含表约束的信息。
  • psql元命令\d+ tablename显示一个分区表的叶子子表的表约束。

Parent topic: 关于GPORCA

示例

这个CREATE TABLE命令创建一个统一分区表。

  1. CREATE TABLE mlp (id int, year int, month int, day int,
  2. region text)
  3. DISTRIBUTED BY (id)
  4. PARTITION BY RANGE ( year)
  5. SUBPARTITION BY LIST (region)
  6. SUBPARTITION TEMPLATE (
  7. SUBPARTITION usa VALUES ( 'usa'),
  8. SUBPARTITION europe VALUES ( 'europe'),
  9. SUBPARTITION asia VALUES ( 'asia'))
  10. ( START ( 2006) END ( 2016) EVERY ( 5));

下面这些是为表mlp创建的子表和分区层次。这个层次由一个包含两个分支的子分区构成。

  1. mlp_1_prt_11
  2. mlp_1_prt_11_2_prt_usa
  3. mlp_1_prt_11_2_prt_europe
  4. mlp_1_prt_11_2_prt_asia
  5. mlp_1_prt_21
  6. mlp_1_prt_21_2_prt_usa
  7. mlp_1_prt_21_2_prt_europe
  8. mlp_1_prt_21_2_prt_asia

该表的层次是统一的,每个分区包含三个子表(子分区)的集合。区域子分区的约束也是统一的,分支表mlp_1_prt_11的子表上的约束集和分支表mlp_1_prt_21的子表的约束相同。

作为一种快速检查,这个查询显示这些分区的约束。

  1. WITH tbl AS (SELECT oid, partitionlevel AS level,
  2. partitiontablename AS part
  3. FROM pg_partitions, pg_class
  4. WHERE tablename = 'mlp' AND partitiontablename=relname
  5. AND partitionlevel=1 )
  6. SELECT tbl.part, consrc
  7. FROM tbl, pg_constraint
  8. WHERE tbl.oid = conrelid ORDER BY consrc;

Note: 注意:您需要修改查询以获得更复杂的分区表。 例如,查询不考虑不同schema中的表名。

consrc列显示子分区上的约束。mlp_1_prt_1中子分区的区域约束集合匹配mlp_1_prt_2中子分区的约束。year的约束继承自父分支表。

  1. part | consrc
  2. --------------------------+------------------------------------
  3. mlp_1_prt_2_2_prt_asia | (region = 'asia'::text)
  4. mlp_1_prt_1_2_prt_asia | (region = 'asia'::text)
  5. mlp_1_prt_2_2_prt_europe | (region = 'europe'::text)
  6. mlp_1_prt_1_2_prt_europe | (region = 'europe'::text)
  7. mlp_1_prt_1_2_prt_usa | (region = 'usa'::text)
  8. mlp_1_prt_2_2_prt_usa | (region = 'usa'::text)
  9. mlp_1_prt_1_2_prt_asia | ((year >= 2006) AND (year < 2011))
  10. mlp_1_prt_1_2_prt_usa | ((year >= 2006) AND (year < 2011))
  11. mlp_1_prt_1_2_prt_europe | ((year >= 2006) AND (year < 2011))
  12. mlp_1_prt_2_2_prt_usa | ((year >= 2011) AND (year < 2016))
  13. mlp_1_prt_2_2_prt_asia | ((year >= 2011) AND (year < 2016))
  14. mlp_1_prt_2_2_prt_europe | ((year >= 2011) AND (year < 2016))
  15. (12 rows)

如果用下面的命令对例子中的分区表增加一个默认分区:

  1. ALTER TABLE mlp ADD DEFAULT PARTITION def

该分区表仍然会是一个统一分区表。为默认分区创建的分支包含三个子表并且子表上的约束集匹配现有的子表约束集。

在上面的示例中,如果删除子分区mlp_1_prt_21_2_prt_asia并为区域canada添加另一个子分区,则约束不再一致。

  1. ALTER TABLE mlp ALTER PARTITION FOR (RANK(2))
  2. DROP PARTITION asia ;
  3. ALTER TABLE mlp ALTER PARTITION FOR (RANK(2))
  4. ADD PARTITION canada VALUES ('canada');

但是,如果将子分区canada添加到原始分区表的mlp_1_prt_21和mlp_1_prt_11,它仍然是统一的分区表。

Note: 只有分区级别的分区集上的约束必须相同。分区的名称可以不同。