PostgreSQL 分区。约束排除未在 SELECT 上启动

标签 postgresql constraints partitioning

我的分区表 SELECT 查询包括所有分区表,即使检查到位并且 constraint_exclusion = on。

插入触发器工作正常,新行被插入到正确的表中。然而,无论我的 WHERE 子句如何,SELECT 都会遍历所有表。

这是我的配置:

constraint_exclusion = on (both in postgresql.conf and also tried with "ALTER DATABASE bigtable SET constraint_exclusion=on;")

主表:

CREATE TABLE bigtable (
    id bigserial NOT NULL,
    userid integer NOT NULL,
    inserttime timestamp with time zone NOT NULL DEFAULT now()
)

子表 1:

CREATE TABLE bigtable_2013_11 (CHECK ( inserttime >= DATE '2013-11-01' AND inserttime < DATE '2013-12-01' )) INHERITS (bigtable);        

子表 2:

CREATE TABLE bigtable_2013_12 (CHECK ( inserttime >= DATE '2013-12-01' AND inserttime < DATE '2014-01-01' )) INHERITS (bigtable);    

存储过程:

CREATE OR REPLACE FUNCTION bigtable_insert_function()
RETURNS TRIGGER AS $$
BEGIN

    IF ( NEW.inserttime >= DATE '2013-11-01' AND NEW.inserttime < DATE '2013-11-01' ) THEN
        INSERT INTO bigtable_2013_11 VALUES (NEW.*);
    ELSEIF (NEW.inserttime >= DATE '2013-12-01' AND NEW.inserttime < DATE '2014-01-01' ) THEN
        INSERT INTO bigtable_2013_12 VALUES (NEW.*);
    ELSE
        RAISE EXCEPTION 'Bigtable insert date is out of range!';
    END IF;

    RETURN NULL;
END;
$$
LANGUAGE plpgsql;

触发器:

CREATE TRIGGER bigtable_insert_trigger BEFORE INSERT ON bigtable FOR EACH ROW EXECUTE PROCEDURE bigtable_insert_function();

这几乎是教科书的设置。插入工作正常:

INSERT INTO bigtable (userid, inserttime) VALUES ('1', now());

上面的插入导致新行仅被正确插入到“bigtable_2013_11”中。

但是我无法让 SELECT 排除不相关的表。所有 SELECT 总是运行在所有表上。我希望在使用以下 SELECT 查询时排除 bigtable_2013_12:

SELECT * FROM bigtable WHERE inserttime >= DATE '2013-11-01'::date AND inserttime < '2013-12-01'::date;

SELECT * FROM bigtable WHERE EXTRACT(MONTH FROM inserttime) = 11 AND EXTRACT (YEAR FROM inserttime) = 2013;

但是结果总是这样:

"Result  (cost=0.00..68.90 rows=17 width=20)"
"  ->  Append  (cost=0.00..68.90 rows=17 width=20)"
"        ->  Seq Scan on bigtable  (cost=0.00..0.00 rows=1 width=20)"
"              Filter: ((inserttime >= '2013-11-02'::date) AND (inserttime < '2013-11-30'::date))"
"        ->  Seq Scan on bigtable_2013_11 bigtable  (cost=0.00..34.45 rows=8 width=20)"
"              Filter: ((inserttime >= '2013-11-02'::date) AND (inserttime < '2013-11-30'::date))"
"        ->  Seq Scan on bigtable_2013_12 bigtable  (cost=0.00..34.45 rows=8 width=20)"
"              Filter: ((inserttime >= '2013-11-02'::date) AND (inserttime < '2013-11-30'::date))"

为什么我的支票没有收到?我没主意了。一切似乎都设置正确。我错过了什么吗?任何帮助将不胜感激。

最佳答案

这里有两件事。一种是序列扫描。另一个是 union all(即追加运算符)。据我所知,后者是完全不可避免的。这就是表继承的工作原理——至少我上次使用它时是这样的。 *但是*,这将我们带到了前者,您可以在该日期字段上添加索引以消除不需要的序列扫描并尽早退出。


或者,可能与日期/时间戳转换有关。

关于PostgreSQL 分区。约束排除未在 SELECT 上启动,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19759947/

相关文章:

postgresql - 两列 : Integer and Boolean 上的 Postgres 唯一约束

python - 满足条件的最近 k 个邻居(python)

postgresql - postgres查询更新小数增量+1

sql - 查询以计算特定列值的出现次数

regex - Postgresql regexp_replace 负向前瞻不起作用

database - 创建分区使用本地时区的时间戳 [Oracle 11g]

mysql - 按 unix 时间戳的 Hive 动态分区

postgresql - 计算表的列中值的出现次数 - postgresql

iOS 如何激活/停用特定大小类(方向)的布局约束?

Mysql 5.5 表分区用户和好友